类成员的可访问范围
在类的定义中,用下列访问范围关键字来说明类成员可被访问的范围:
- private:私有成员,只能在成员函数内访问。
- public:公有成员,可以在任何地方访问。
- protected:保护成员。
如果某个成员前面没有上述关键字,则缺省的被默认为是私有成员。
1 | class Man{ |
在类的成员函数内部,能够访问:
- 当前对象的全部属性、函数;
- 同类其他对象的全部属性、函数。
在类的成员函数以外的地方,只能够访问该类对象的公有成员。一个例子如下:
1 | class CEmployee{ |
设置私有成员的机制,叫“隐藏”
“隐藏”的目的是强制对成员变量的访问一定要通过成员函数进行,那么以后成员变量的类型属性修改后,只需要更改成员函数即可。否则,所有访问成员变量的语句都需要修改。
“隐藏”的作用
如果将上述的程序移植到内存紧张的手持设备上,希望将szName改为szName[5],若szName不是私有,那么就要找出所有类似于 strcpy(e.szName,”Tom1234567899”); 这样的语句进行修改,以防止数组越界。这样做很麻烦。
如果将szName变为私有,那么程序中就不可能再出现strcpy(e.szName,”Tom1234567899”);所有对szName的访问都是通过成员函数来进行,比如:
e.setName(“Tom1234567899”);
那么,就算szName改短了,上面的语句也不需要找出来修改,只要改setName成员函数,在里面确保不越界就可以了。
用struct定义类
1 | struct CEmployee{ |
和class的唯一区别是,就是未说明是公有还是私有的成员,就是公有。
成员函数的重载和参数缺省
成员函数也可以重载。
成员函数可以带缺省参数。
1 |
|
1 | void Location::init(int X,int Y){ |
1 | int main() { |
输出结果为5
注意:使用缺省参数要注意避免有函数重载时的二义性
1 | class Location { |
构造函数
基本概念
是成员函数的一种:1.名字与类名相同,可以有参数,不能有返回值(void也不行);2.作用是对对象进行初始化,如给成员变量赋初值;3.如果定义类时没写构造函数,则编译器生成一个默认的无参数的构造函数:默认构造函数无参数,不做任何操作。
如果定义了构造函数,则编译器不生成默认的无参数的构造函数。
对象生成时构造函数自动被调用。对象一旦生成,就再也不能在其上执行构造函数。
一个类可以有多个构造函数。
为什么需要构造函数:
1.构造函数执行必要的初始化工作,有了构造函数,就不必专门再写初始化函数,也不用忘记调用初始化函数。
2.有时对象没被初始化就使用,会导致程序出错。
1 | class Complex{ |
1 |
|
可以有多个构造函数,参数个数或类型不同
1 | class Complex{ |
构造函数最好是public的,private的构造函数不能直接用来初始化对象
1 | class CSample{ |
构造函数在数组中的使用
1 | class CSample{ |
结果输出:
1 | 输出: |
1 | class Test{ |
复制构造函数
基本概念
只有一个参数,即对同类对象的引用。
形如 X::X(X&)或X::X(const X &),二者选一,后者能以常量对象作为参数
如果没有定义复制构造函数,那么编译器生成默认复制构造函数。默认的复制构造函数完成复制功能。
1 | class Complex{ |
如果定义的自己的复制构造函数,则默认的复制构造函数不存在。
1 | class Complex{ |
不允许有形如X::X(X)的构造函数。
1 | class CSample{ |
复制构造函数起作用的三种情况
1)当用一个对象去初始化同类的另一个对象时。
Complex c2(c1);
Complex c2=c1;//初始化语句,非赋值语句
2)如果某函数有一个参数是类A的对象,那么该函数被调用时,类A的复制构造函数将被调用。
1 | class A{ |
程序输出结果为:Copy constructor called
3)如果函数的返回值是类A的对象时,则函数返回时,A的复制构造函数被调用:
1 | class A |
注意:对象间赋值并不导致复制构造函数被调用
1 | class CMyclass{ |
常量引用参数的使用
1 | void fun(Cmyclass obj_){ |
这样的函数,调用时生成形参会引发复制构造函数的调用,开销比较大。
所以可以考虑使用CMyclass &引用类型作为参数。
如果希望确保实参的值在函数中不应被改变,可以加上const关键字:
1 | void fun(const CMyclass &obj){ |
类型转换构造函数
定义转换构造函数的目的是为了实现类型的自动转换。
只有一个参数,而且不是复制构造函数的构造函数,一般就可以看做是转换构造函数。
当需要的时候,编译系统会自动调用转换构造函数,建立一个无名的临时对象(或临时变量)。
1 | class Complex{ |
析构函数
析构函数
1.名字与类名相同,在前面加上‘~’,没有参数和返回值,一个类最多只有一个析构函数。
2.析构函数对象消亡即自动被调用。可以定义析构函数来在对象消亡前后做善后工作,比如释放分配的空间等。
3.如果定义类时没写析构函数,则编译器生成缺省析构函数。缺省析构函数什么也不做。
4.如果定义了析构函数,则编译器不生成缺省析构函数。
一个示例如下:
1 | class String{ |
对象数组生命期结束时,对象数组的每个元素的析构函数都会被调用。
1 | class Ctest{ |
delete 运算导致析构函数调用。
1 | Ctest *pTest; |
若new一个对象数组,那么用delete释放时应该写[]。否则只delete一个对象(调用一次析构函数)
析构函数在对象作为函数返回值后被调用
1 | class CMyclass{ |
析构函数和析构函数什么时候被调用
1 | class Demo{ |
关于复制构造函数和析构函数的又一个例子
1 | class CMyclass{ |