员函数调整客户的银行借贷(例如BanLAccount类的private数据成员)。
类通常提供public成员函数,让类的客户设置(写入)或读取(取得)private数据成员的值。这些函数通常称为get和set。更具体地说,设置数据成员interestRate的成员函数通常称为setInterestRate,读取数据成员IntersetRate的值通常称为getInterestRate。读取函数也称为“查询”函数。
提供get和set函数与指定数据成员为public同样重要,这是C++语言在软件工程中的另一优势。如果数据成员为public,则程序中的任何函数可以随意读取和写入这个数据成员。如果数据成员为private.则public get函数可以让其他函数读取数据,而且数据的显示和格式化也可以用get函数控制。public set函数通常用于检查数据成员的修改,保证新值是适当的数据项目。例如,如果想把一个月的日期号数设置为37会被禁止,将人的身高设置为负值也会被禁止,将数字量设置为字母值也会被拒绝,将一个人的成绩设置为185分(取百分制时)同样也会被拒绝等等。
指定private数据成员并通过public成员函数控制这些数据成员的访问(特别是写入访问)可以保证数据的完整性。
指定private数据成员并不能自动保证数据完整性,程序员还要提供验证检查。但C++提供了让程序员方便地设计更好的程序的框架。
设置private数据值的成员函数应验证所要新值是否正确,如果不正确,则set数应将Privte数据成员设置为相应的一致状态。
试图要对数据成员指定无效值时,应当提醒类客户。类的set函数常写成返回一个值,表示试图对数据成员指定无效值。这样就使类的客户可以测试set函数的返回值,确定其操作的对象是否为有效对象,并在对象无效时采取相应操作。
图6.1O将Time类扩展成包括private数据成员hour、minute和second的get和set函数。set函数严格控制数据成员的设置。如果想把数据成员设置为无效值,则会把数据成员设置为0(从而使数据成员保持一致状态)。每个get函数只是返回相应数据成员的值。程序首先用set函数设置Time对象t的private数据成员为有效值,接着用get函数读取这个值以便输出。然后set函数要将hour和second成员设置为无效值并将minute成员设置为有效值,并用get函数读取这个值以便输出。输出表明,无效值使得数据成员设置为0。最后,程序将时间设置为11:58:00并用函数incrementMinutes增加3分钟。函数incrementMinutes是个非成员函数,它调用get和set成员函数增加minute成员的值。尽管这样的方法实现了所需的功能,但是多次函数调用降低了程序的性能。下一章将介绍用友元函数消除多次函数调用的性能负担。
构造函数可以调用类的其他成员函数,如set和get函数,但由于构造函数初始化对象,因此数据成员可能还处于不一致状态。数据成员在初始化之前使用可能造成逻辑错误。
// Fig. 6.10: time3.h // Declaration of the Time class. // preprocessor directives that // prevent multiple inclusions of header file #ifndef TIME3_H #define TIME3_H class Time { public: Time( int = 0, int = 0, int= 0 ); // constructor // set functions void setTime( int, int, int ); // set hour, minute, se void setHour( iht ); // set hour void setMinute( int ); // set minute void setSecond( int ); // set second // get functions int getHourO; // return hour int getMinute(); // return minute int getSecond(); // return second void printMilitary(); // output military time void printStandard(); // output standard time private: int hour; // 0 - 23 int minute; // 0 - 59 int second; // 0 - 59 } #endif // Fig. 6.10: time3.cpp // Member function defintions for Time class #include "time3.h" #include < iostream.h> // Constructor function to initialize private data. // Default values are 0 (see class definition). Time::Time( int hr, int min, int sec ) { setTime( hr, min, sec ); } // Set the values of hour, minute, and second. void Time::setTime(int h,int m,int s) { setHour( h ); setMinute( m ); setSecond( s ); } // Set the hour value void Time::setHour(int h) {hour = (h>0 && h <24 )? h: 0;} // Set the minute value void Time::setMinute( int m ) { minute = ( m >= 0 && m 60 ) ? m : 0; } // Set the second value void Time::setSecond( int s { second = ( s >= 0 && s < 60 ) ? s : 0; } // Get the hour value int Time::getHour() { return hour;} // Get the minute value int Time::getMinute() { return minute; } // Get the second value int Time::getSecond() { return second; } // Print time is military format void Time::printMilitary() { cout << ( hour < 10 ? "0" : "" ) << hour << ":" << ( minute < 10 ? "0" : "" ) << minute; } // Print time in standard format void Time::printStandard{) { cout << ( { hour == 0 II hour == 12 ) ? 12 : hour % 12 ) << ":" << ( minute < 10 ? "0" : "" ) << minute << ":" << ( second < 10 ? "0" : "" ) << second << ( hour < 12 ? "AM" : "PM" ); } // Fig. 6.10: fig06_lO.cpp #include < iostream.h> // Demonstrating the Time class ser and get functions #include< iostream.h> #include "time3.h" void incrementMinutes( Time &, const iht ); int main() { Time t; t.setHour{ 17 ); t.setMinute( 34 ); t.setSecond( 25 ); cout << "Result of setting all valid values:\n; << our: << t.getHour() << " Minute: " << t.getMinute() <<" Second: "<< t.getSecond(); t.setHour( 234 ); // invalid hour set to 0 t.setMinute( 43 ); t.setSecond( 6373 ); // invalid second set to 0 cout << "\n\nResult of attempting to set invalid hour and" << "second:\m Hour: "<< t.getHour() <<" Minute: "<< t.getMinute() <<" Second: "<< t.getSecond() << "\n\n"; t.setTime( 11, 58, 0 ); incrementMinutes( t, 3 ); return 0; } void incrementMinutes(Time &tt, const int count) { cout << "Incrementing minute" << count << "times:\nStart time: "; tt.priatStandard(); for (int i = 0; i < count; i++ ) { tt.setMinute( (tt.getMinute() + 1 ) % 60); if (tt.getMinute() == 0 ) tt.setHour( ( tt.getHour() + 1 ) % 24); cout << "\nminute + 1: "; tt.printStandard(); } cout << endl; }
输出结果:
Result of setting all valid values:
Hour: 17 Minute: 34 Second: 25
Result of attempting to set inv]id hour and second:
Hour: 0 Minute: 43 Second: 0
Incrementing minute 3 times:
Start time: 11:58:00 AM
minute + 1; 11:59:00 AM
mioute + 1:12:00:00 PM
minute + 1:12:01:00 PM
图 6.10 使用set和get函数
从软件工程角度看,使用set函数非常重要,因为它们可以进行有效性检查。set和get函数还有其他重要的软件工程优势。
通过set和get函数访问private数据不仅能防止数据成员接受无效值,而且还使类的客户J需要考虑数据成员的表达方式。这样,如果数据表达方式因故改变(通常是为了减少所需存储量或提高性能),只要成员函数提 供的接口不变,那么只需改变成员函数而不必改变客户,但客户可能需要重新编译。
对象的引用是对象名的别名,因此可以放在赋值浯句左边,在这种情况中,引用可以成为可接收赋值的左值。要使用这种功能,就要让类的public成员函数返回对该类private数据成员的非const引用。
图6.11用简化的Time类演示如何返回private数据成员的引用。调用badSetHour函数返回的引用作为private数据成员hour的别名。函数调用可以按任何使用private数据成员的方式使用,包括作为赋值语句的左值。
不要让类的public成员函数返回对该类private数据成员的非const引用(或指针),返回这种引用会破坏类的封装。
// Fig. 6.11: time4.h
// Declaration of the Time class.
// Member functions defined in time4.cpp
// Fig. 6.11: time4.h
// Declaration of the Time class.
// Member functions defined in time4.cpp
// preprocessor directives that
// prevent multiple inclusions of header file
#ifndef TIME4_H
#define TIME4_H
class Time {
public
Time( int = 0, int = 0, int = 0 );
void setTime( int, int, int );
int getHour();
int &badSetHour( int ); // DANGEROUS reference return
private:
int hour;
int minute;
int second;
};
#endif
// Fig. 6.11: time4.cpp
// Member function definitions for Time class.
#include "time4.h"
#include <iostream.h>
// Constructor function to initialize private data.
// Calls member function setTime to set variables.
// Default values are 0 (see class definition).
Time::Time( int hr, int min, int sec )
{ setTime( hr, min, sec ); }
// Set the values of hour, minute, and second.
void Time::setTime( int h, int m, int s )
{
hour = ( h >= o && h < 24 ) ? h : 0;
minute ( m >= 0 && m < 60 ) ? m : 0;
second = ( s >= 0 && s < 60 ) ? s : 0;
}
// Get the hour value
int Time::getHour() { return hour; ]
// POOR PROGRAMMING PRACTICE:
// Returning a reference to a private data member.
int &Time::badSetHour( int hh )
{
hour = ( hh >= 0 && hh < 24 ) ? hh : 0;
return hour; // DANGEROUS reference return
}
// Fig. 6.11: fig06_11.cpp
// Demonstrating a public member function that
// returns a reference to a private data member.
// Time class has been triced for this example.
#include <iostream.h>
#include "time4.h"
int main()
{
Time t;
int &hourRef = t.badSetHour( 20 );
cout << "Hour before modification: "<< hourRef;
hourRef = 30; // modification with invalid value
cout << "\nHour after modification: "<< t.getHour();
// Dangerous: Function call that returns
// a reference can be used as an lvalue!
t.badSetHour(12) = 74;
cout << "\n\n*******************************************\n"
<< "POOR PROGRkMMING PRACTICE!!!!!!!!\n"
<< "badSetHour as an lvalue, Hour:"
<< t.getHour()
<< "\n*************************************** << endl;
return 0;
}
// preprocessor directives that
// prevent multiple inclusions of header file
#ifndef TIME4_H
#define TIME4_H
class Time {
public
Time( int = 0, int = 0, int = 0 );
void setTime( int, int, int );
int getHour();
int &badSetHour( int ); // DANGEROUS reference return
private:
int hour;
int minute;
int second;
};
#endif
// Fig. 6.11: time4.cpp
// Member function definitions for Time class.
#include "time4.h"
#include <iostream.h>
// Constructor function to initialize private data.
// Calls member function setTime to set variables.
// Default values are 0 (see class definition).
Time::Time( int hr, int min, int sec )
{ setTime( hr, min, sec ); }
// Set the values of hour, minute, and second.
void Time::setTime( int h, int m, int s )
{
hour = ( h >= o && h < 24 ) ? h : 0;
minute ( m >= 0 && m < 60 ) ? m : 0;
second = ( s >= 0 && s < 60 ) ? s : 0;
}
// Get the hour value
int Time::getHour() { return hour; ]
// POOR PROGRAMMING PRACTICE:
// Returning a reference to a private data member.
int &Time::badSetHour( int hh )
{
hour = ( hh >= 0 && hh < 24 ) ? hh : 0;
return hour; // DANGEROUS reference return
}
// Fig. 6.11: fig06_11.cpp
// Demonstrating a public member function that
// returns a reference to a private data member.
// Time class has been triced for this example.
#include <iostream.h>
#include "time4.h"
int main()
{
Time t;
int &hourRef = t.badSetHour( 20 );
cout << "Hour before modification: "<< hourRef;
hourRef = 30; // modification with invalid value
cout << "\nHour after modification: "<< t.getHour();
// Dangerous: Function call that returns
// a reference can be used as an lvalue!
t.badSetHour(12) = 74;
cout << "\n\n*******************************************\n"
<< "POOR PROGRkMMING PRACTICE!!!!!!!!\n"
<< "badSetHour as an lvalue, Hour:"
<< t.getHour()
<< "\n*************************************** << endl;
return 0;
}
输出结果:
Hour before modification: 20
Hour after modification: 30
*********************************
POOR PROGRAMMING PRACTICE!!!!!!!!
badSetHour as an lvalue, Hour: 74
*********************************
图 6.11 返回对private数据成员引用
程序首先声明Time对象t和引用hourRef(把调用t.badSetHour(20)返回的引用赋给hourRef)。程序显示别名hourRef的值。然后用这个别名设置hour的值为30(无效值)并再次显示该值。最后,用函数调用本身作为左值并赋值74(另一无效值),再次显示该值。