9.3拷贝初始化构造函数 class name ,class name(class name& func body; 拷贝初始化构造函数为利用一个已存在的对象来创建另一个 对象提供了接口。例如: Person per1("张三",21,'m"),per2(er1); 其中,对象per2就是利用对象per来初始化的。因此这两 个对象具有其值完全相同的数据成员 注意:前边并没有为 Person类定义拷贝初始化构造函数,创 建per2时调用的是什么呢?是缺省的拷贝初始化构造函数。 若类中没有显式地定义拷贝初始化构造函数,则系统将自动 生成一个函数体为空的缺省的拷贝初始化构造函数
9.3 拷贝初始化构造函数 <class_name :: >class_name(class_name&) { func_body; } 拷贝初始化构造函数为利用一个已存在的对象来创建另一个 对象提供了接口。例如: Person per1("张三", 21, 'm'), per2(per1); 其中,对象 per2 就是利用对象 per1 来初始化的。因此这两 个对象具有其值完全相同的数据成员。 注意:前边并没有为 Person 类定义拷贝初始化构造函数,创 建 per2 时调用的是什么呢?是缺省的拷贝初始化构造函数。 若类中没有显式地定义拷贝初始化构造函数,则系统将自动 生成一个函数体为空的缺省的拷贝初始化构造函数
然而,对于本例而言,利用缺省的拷贝初始化构造函数却会带 来一个严重和问题:当对象per1生存期结束时,其析构函数 会释放对象中数据成员 pName所指的动态内存并释放整个对 象所占的内存。而当per2消亡时,析构函数在释放整个对象 所占内存前首先要释放per2 pName所指的动态内存,而这 块内存已经在撤消per1时释放掉了。这将导致一个“运行 时间错误(Run- Time error) 对于那些涉及到动态内存分配的类,必须显式地定义拷贝初始 化构造函数。 从拷贝初始化构造函数的一般形式可以看出,该函数是不允许 重载的,因此显式定义的拷贝初始化构造函数将覆盖缺省的拷 贝初始化构造函数
然而,对于本例而言,利用缺省的拷贝初始化构造函数却会带 来一个严重和问题:当对象 per1 生存期结束时,其析构函数 会释放对象中数据成员 pName 所指的动态内存并释放整个对 象所占的内存。而当 per2 消亡时,析构函数在释放整个对象 所占内存前首先要释放 per2.pName 所指的动态内存,而这 一块内存已经在撤消 per1 时释放掉了。这将导致一个“运行 时间错误(Run-Time Error)”。 对于那些涉及到动态内存分配的类,必须显式地定义拷贝初始 化构造函数。 从拷贝初始化构造函数的一般形式可以看出,该函数是不允许 重载的,因此显式定义的拷贝初始化构造函数将覆盖缺省的拷 贝初始化构造函数
∥在 PERSON.H中添加公有的成员函数原型: Person(Person&) ∥在 PERSON. CPP中添加函数定义 Person :: Person(Person& per) pName new char[strlen(per pName)+ 1] strcpy(pName, per pName)N uAge per uAge uSex = per uSex 经过这样的定义,上述per2在创建时将为自身的 pName申 请一块动态内存,并将per1中的 pName所指的字符串复制 到该动态内存中。当任一对象消亡时,释放的均为自身的动 态内存,从而防止了相应的运行时间错误发生
// 在 PERSON.H 中添加公有的成员函数原型: Person(Person&); // 在 PERSON.CPP 中添加函数定义: Person :: Person(Person& per) { pName = new char[strlen(per.pName) + 1]; strcpy(pName, per.pName); uAge = per.uAge; uSex = per.uSex; } 经过这样的定义,上述 per2 在创建时将为自身的 pName 申 请一块动态内存,并将 per1 中的 pName 所指的字符串复制 到该动态内存中。当任一对象消亡时,释放的均为自身的动 态内存,从而防止了相应的运行时间错误发生
94-构造函数与对象成 结构变量可以用作另一个结构的成员。同理,对象也可以作为 另一个类的数据成员。例: include person. h class Family 三口之家类 private: Person Father, Person Mother: Person Child
9.4 构造函数与对象成员 结构变量可以用作另一个结构的成员。同理,对象也可以作为 另一个类的数据成员。例: #include "person.h" class Family { // 三口之家类 private: Person Father; Person Mother; Person Child; //… };
为了初始化类中的对象成员,类中应当定义具有以下一般形式 的构造函数: class name(args): C(arg,), C2(arg2). Cn(argn) func bodyi 其中:各C分别为类中的对象成员名;ags为多个形式参数; 而各arg则为args的子集 例 Family(Person pa, Person ma, Person kid) Father(pa), Mother(ma), Child(kid) 这里,冒号后边的各项就分别调用了 Person类中的拷贝初始 化构造函数。因此,显式定义 Person类中的拷贝初始化构造 函数就显得更为重要
为了初始化类中的对象成员,类中应当定义具有以下一般形式 的构造函数: class_name(args) : c1(arg1), c2(arg2)…cn(argn) { func_body; } 其中:各 ci 分别为类中的对象成员名;args 为多个形式参数; 而各 argi 则为 args 的子集。 例: Family(Person pa, Person ma, Person kid) : Father(pa), Mother(ma), Child(kid) {} 这里,冒号后边的各项就分别调用了 Person 类中的拷贝初始 化构造函数。因此,显式定义 Person 类中的拷贝初始化构造 函数就显得更为重要