继承与派生(三)
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
5.3.1派生类成员标识与访问:同名隐藏
当派生类与基类中有相同成员时:
●若未强行指名,则通过派生类对象使用的是派生类中的同名成员。
●如要通过派生类对象访问基类中被覆盖的同名成员,应使用基类名限定。
示例:
#include
using namespace std;
class B1 //声明基类B1
{ public: //外部接口
int nV;
void fun() {cout<<"B1的成员函数fun"< }; class B2 //声明基类B2 { public: //外部接口 int nV; void fun(){cout<<"B2的成员函数fun"< }; class D1: public B1, public B2 { public: int nV; //同名数据成员 void fun(){cout<<"D1的成员函数fun"< }; void main() { D1 d1; d1.nV=1; //对象名.成员名标识, 访问D1类成员 d1.fun(); d1.B1::nV=2; //作用域分辨符标识, 访问基类B1成员 d1.B1::fun(); d1.B2::nV=3; //作用域分辨符标识, 访问基类B2成员 d1.B2::fun(); } 5.3.2 派生类成员标识与访问:二义性问题 ●在多继承时,基类与派生类之间,或基类之间出现同名成员时,将出现访问时的二 义性(不确定性)——采用虚函数或同名隐藏规则来解决。 ●当派生类从多个基类派生,而这些基类又从同一个基类派生,则在访问此共同基类 中的成员时,将产生二义性——采用虚基类来解决。 示例: #include using namespace std; class A { public: void f(); }; class B { public: void f(); void g(); }; class C: public A, public B { public: void g(); void h(); }; void main() { C c1; c1.g();//正确,无二义性(同名覆盖) c1.f();//错误,具有二义性. //解决方法一:用类名来限定c1.A::f() 或c1.B::f() //解决方法二:【同名覆盖】在C中声明一个同名成员函数f(),f()再根据需要调用A::f()或B::f() } 5.3.3 虚基类 ●虚基类的引入:用于有共同基类的场合 ●声明:以virtual修饰说明基类,例:class B1:virtual public B ●作用: 1、主要用来解决多继承时可能发生的对同一基类继承多次而产生的二义性问题. 2、为最远的派生类提供唯一的基类成员 ●注意:在第一级继承时就要将共同基类设计为虚基类。 5.3.3.1虚基类示例1: #include using namespace std; class B { public: int b; }; class B1:virtual public B { private: int b1; }; class B2:virtual public B { private: int b2; }; class C:public B1, public B2 { private: float d; }; void main(void) { C cobj; cobj.b; //此处的访问是正确的 } 上述4个类的继承关系(B,B1,B2,C)的继承关系和对象存储结构如图 5.3.3.2虚基类示例2: #include using namespace std; class B0 //声明基类B0 { public: //外部接口 int nV; void fun(){cout<<"B0的成员函数fun"< }; class B1: virtual public B0 //B0为虚基类,派生B1类 { public: //新增外部接口 int nV1; //void fun(){cout<<"B1的成员函数fun "< }; class B2: virtual public B0 //B0为虚基类,派生B2类 { public: //新增外部接口 int nV2; }; class D1: public B1, public B2 //派生类D1声明 { public: //新增外部接口 int nVd; void fund(){cout<<"D1的成员函数fund"< }; void main() { D1 d1; //声明D1类对象d1 d1.nV=2; //使用最远基类成员 d1.fun();//此时调用的是最远基类成员函数;若将B1中的fun()的注释去掉呢?}