第15章 面向对象编程(6)
15.3 转换与继承
基类类型对象既可以作为独立对象存在,也可以作为派生类对象的一部分而存在,因此,一个基类对象可能是也可能不是一个派生类对象的部分,结果,没有从基类引用(或基类指针)到派生类引用(或派生类指针)的(自动)转换。
相对于引用或指针而言,对象转换的情况更为复杂。虽然一般可以使用派生类型的对象对基类类型的对象进行初始化或赋值,但,没有从派生类型对象到基类类型对象的直接转换。
15.3.1 派生类到基类的转换
如果有一个派生类型的对象,则可以使用它的地址对基类类型的指针进行赋值或初始化。同样,可以使用派生类型的引用或对象初始化基类类型的引用。严格说来,对对象没有类似转换。编译器不会自动将派生类型对象转换为基类类型对象。
但是,一般可以使用派生类型对象对基类对象进行赋值或初始化。对对象进行初始化和/或赋值以及可以自动转换引用或指针。
1. 引用转换不同于转换对象
可以将派生类型的对象传给希望接受基类引用的函数。也许会因此认为对象进行转换,但是事实并非如此。将对象传给希望接受引用的函数时,引用直接绑定到该对象,虽然看起来在传递对象,实际上是该对象的引用,对象本身未被复制,并且,转换不会在任何方面改变派生类型对象,该对象仍是派生类型对象。
将派生类对象传给希望接受基类类型对象(而不是引用)的函数时,情况完全不同。在这种情况下,形参的类型是固定的——在编译时和运行时形参都是基类类型对象。如果用派生类型对象调用这样的函数,则该派生类对象的基类部分被复制到形参。
一个是将派生类对象转换为基类类型引用,一个是用派生类对象对基类对象进行初始化或赋值,理解它们之间的区别很重要。
2. 用派生类对象对基类对象进行初始化或赋值
对基类对象进行初始化或赋值,实际上实在调用函数:初始化时调用构造函数,赋值时调用赋值操作符。
用派生类对象对基类对象进行初始化或赋值时,有两种可能性。第一种(虽然不大可能的)可能性是,基类可能显式定义了将派生类型对象复制或赋值给基类对象的含义,这可以通过定义适当的构造函数或赋值操作符实现。
基类一般(显式或隐式地)定义自己的复制构造函数或赋值操作符,这些成员接受一个形参,该形参是基类类型的(const)引用。因为存在从派生类引用到基类引用的转换,这些复制控制成员可用于从派生类对象对基类对象进行初始化或赋值。
3. 派生类到基类转换的可访问性
像继承的成员函数一样,从派生类到基类的转换可能是也可能不是可访问的。转换是否可访问取决于在派生类的派生列表中指定的访问标号。
要确定到基类的转换是否可访问,可以考虑基类的public成员是否可访问,如果可以,转换是可访问的,否则,转换是不可访问的。
如果是public继承,则用户代码和后代类都可以使用派生类到基类的转换。如果类是使用private或protected继承派生的,则用户代码不能将派生类型对象转换为基类对象。如果是private继承,则从private继承类派生的类不能转换为基类。如果是protected继承,则后续派生类的成员可以转换为基类类型。
无论是什么派生访问标号,派生类本省都可以访问基类的public成员,因此,派生类本身的成员和友元总是可以访问派生类到基类的转换。
class Base{
protected:
int i;
public:
int j;
private:
int k;
};
class Child:protected Base{};
class Child1:public Base{};
class Child2:public Child{
void f2(){
i;
Base b(*this);
b.j;
}
};
class Child3:private Base{
void f1(){
i;
}
};
class Child4:public Child3{
void f2(){
//nothing accessiable...
}
};
Child1 c1=Child1();
Base b(c1);
摘自 xufei96的专栏
补充:软件开发 , C++ ,