面向对象编程
--转换与继承、复制控制与继承
I.转换与继承
引言:
由于每个派生类对象都包含一个基类部分,因此可以像使用基类对象一样在派生类对象上执行操作。
对于指针/引用,可以将派生类对象的指针/引用转换为基类子对象的指针/引用。
基类类型对象既可以作为独立对象存在,也可以作为派生类对象的一部分而存在,因此,一个基类对象可能是也可能不是一个派生类对象的部分,因此,没有从基类引用(或基类指针)到派生类引用(或派生类指针)的(自动)转换。
关于对象类型,虽然一般可以使用派生类型的对象对基类类型的对象进行初始化或赋值,但,没有从派生类型对象到基类类型对象的直接转换。
一、派生类到基类的转换
如果有一个派生类型的对象,则可以使用它的地址对基类类型的指针进行赋值或初始化。同样,可以使用派生类型的引用或对象初始化基类类型的引用。但是,对象没有类似转换。编译器不会自动将派生类型对象转换为基类类型对象。
但是,一般可以使用派生类型对象对基类对象进行赋值或初始化。对对象进行初始化和/或赋值以及可以自动转换引用或指针。
1、引用转换不同于转换对象
关于引用转换:将对象传给希望接受引用的函数时,引用直接绑定到该对象,虽然看起来在传递对象,实际上实参是该对象的引用,对象本身未被复制,并且,转换不会在任何方面改变派生类型对象,该对象仍是派生类型对象。
关于对象转换:将派生类对象传给希望接受基类类型对象(而不是引用)的函数时,形参的类型是固定的――在编译时和运行时形参都是基类类型对象。如果用派生类型对象调用这样的函数,则该派生类对象的基类部分被复制到形参。
小结:一个是派生类对象转换为基类类型引用,一个是用派生类对象对基类对象进行初始化或赋值。
2、用派生类对象对基类对象进行初始化或赋值
对基类对象进行初始化或赋值,实际上是在调用函数:初始化时调用构造函数,赋值时调用赋值操作符。
基类一般(显式或隐式地)定义自己的复制构造函数和赋值操作符,这些成员接受一个形参,该形参是基类类型的(const)引用。因为存在从派生类引用到基类引用的转换,这些复制控制成员可用于从派生类对象对基类对象进行初始化或赋值:
Item_base item;
Bulk_item bulk;
Item_base item1(bulk); //调用Item_base的复制构造函数
item = bulk; //调用Item_base的赋值操作符
用Bulk_item类型的对象调用Item_base类的复制构造函数或赋值操作符时,将发生下列步骤:
1)将Bulk_item对象转换为Item_base引用,这仅仅意味着将一个Item_base引用绑定到Bulk_item对象。
2)将该引用作为实参传给复制构造函数或赋值操作符。
3)那些操作符使用Bulk_item的 Item_base部分分别对调用构造函数或赋值的 Item_base对象的成员进行初始化或赋值。
4)一旦操作符执行完毕,对象即为Item_base。它包含Bulk_item的 Item_base部分的副本,但实参的Bulk_item部分被忽略。
在这种情况下,bulk的Bulk_item部分在对item进行初始化或赋值时被“切掉”了。Item_base对象只包含基类中定义的成员,不包含由任意派生类型定义的成员,Item_base对象中没有派生类成员的存储空间。
3、派生类到基类转换的可访问性
像继承的成员函数一样,从派生类到基类的转换可能也可能不是可访问的。转换是否可以访问取决于在派生类的派生列表中指定的访问标号。
【提示:】要确定到基类的转换是否可访问,可以考虑基类的public成员是否可访问,如果可以,则转换是可访问的,否则,转换是不可访问的!
1)public继承:用户代码和后代类都可以使用派生类到基类的转换。
2)private或 protected继承:则用户代码不能将派生类型对象转换为基类对象。如果是 private继承,则从private继承类派生的类不能转换为基类。如果是protected继承,则后续派生类的成员可以转换为基类类型。
无论是什么派生访问标号,派生类本身都可以访问基类的public[protected]成员,因此,派生类本身的成员和友元总是可以访问派生类到基类的转换。
二、基类到派生类的转换
从基类到派生类的自动转换是不存在的:
Item_base base;
Bulk_item *bulkP = &base; //Error
Bulk_item &bulkRef = base; //Error
Bulk_item bulk = base; //Error
没有从基类类型到派生类型的(自动)转换,原因在于基类对象只能是基类对象,它不能包含派生类对象的成员。如果允许基类对象给派生类类型对象赋值,那么就可以试图使用派生类对象访问不存在的成员!
更有甚者:当基类指针或引用实际绑定到派生类对象时,从基类到派生类的转换也存在限制:
Bulk_item bulk;
Item_base *itemP = &bulk;
Bulk_item *bulkP = itemP; //Error
编译器在编译时无法知道特定转换在运行时实际上是安全的。编译器确定转换是否合法,只看指针或引用的静态类型。
在这些情况下,如果用户知道从基类到派生类的转换是安全的,就可以使用static_cast强制编译器进行转换。或者,可以用dynamic_cast申请在运行时进行检查。
Bulk_item bulk;
Item_base *itemP = &bulk;
Bulk_item *bulkP = static_cast
(itemP); //OK
Bulk_item *bulkP = dynamic_cast
(itemP); //OK
II.复制控制与继承
引言:
当构造、复制、赋值和撤销派生类对象时,也会构造、复制、赋值和撤销基类子对象!
构造函数和复制控制成员不能继承,每个类定义自己的构造函数和复制控制成员。像任何类一样,如果类不定义自己的默认构造函数和复制控制成员,则编译器将使用合成版本!
一、基类构造函数和复制控制
本身不是派生类的基类,其构造函数和复制控制基本上不受继承影响:
Item_base(const std::string &book = "",
double sales_price = 0.0):
isbn(book),price(sales_price) {}
构造函数可以设置成为protected或private,某些类需要只希望派生类使用的特殊构造函数,则将够函数设置成为protected。
二、派生类构造函数
每个派生类构造函数出了初始化自己的数据成员,还要初始化基类!
1、合成的派生类默认构造函数
派生类的合成默认构造函数与非派生类的构造函数只有一点不同:除了初始化派生类的数据成员之外,它还需要初始化派生类对象的基类部分[其实是先初始化基类部分的,一定要掌握好初始化顺序]!
对于Bulk_item类,合成的默认构造函数执行顺序:
1)调用Item_base的默认构造函数;[基类]
2)用常规变量初始化规则初始化Bulk_item的成员,也就是说,qty和discount成员是未