C语言中多重继承如何避免二义性原理的实例应用有哪些?

2026-04-19 00:352阅读0评论SEO教程
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计745个文字,预计阅读时间需要3分钟。

C语言中多重继承如何避免二义性原理的实例应用有哪些?

在派生类中对基类成员访问应该是唯一的,但在多继承时可能会出现对基类成员访问不一致的情况,这就是C++多继承中的二义性问题。有两种多继承的情况会导致二义性:一是如果两个基类有同名的成员,二是如果派生类通过两个基类继承同一个成员。

在派生类中对基类成员访问应该是唯一的,但是在多继承时,可能会导致对基类某成员访问出现不一致的情况,这就是C++多继承中的二义性。

有两种继承的情况会产生多义性

一、如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的成员变量进行访问时,可能产生二义性,继承关系如下图所示:

#include <iostream> using namespace std; class A{ public: int a; }; class B1 : public A{ public: int b1; }; class B2 : public A{ public: int b2; }; class C : public B1, public B2{ public: int c; }; int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c = 300; c1.a = 500; //报错,不能确定成员变量a具体在那个类 cout << "end..." << endl; system("pause"); return 0; }

解决方法:虚继承

C语言中多重继承如何避免二义性原理的实例应用有哪些?

注意:C++编译系统在实例化C类时,只会将虚基类A的构造函数调用一次,忽略虚基类的其他派生类(class B1,class B2)对虚继承的构造函数的调用,从而保证了虚基类的数据成员不会被多次初始化。

在虚基类A中有一个虚指针指向一个虚表,虚表中记录了虚基类与本类的地址偏移,通过这个地址偏移可以找到虚基类的成员变量a的地址

#include <iostream> using namespace std; class A{ public: int a; }; class B1 : virtual public A{ public: int b1; }; class B2 : virtual public A{ public: int b2; }; class C : public B1, public B2{ public: int c1; }; int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c1 = 300; c1.a = 500; //虚继承使得成员变量a只有一份拷贝,通过虚指针可以确定地址 cout << "end..." << endl; system("pause"); return 0; }

二、一个派生类同时继承两个基类,这两个基类存在相同的成员函数

#include<iostream> using namespace std; class Base1 { public: void fun() { cout << "I am base-1 " << endl; }; }; class Base2 { public: void fun() { cout << "I am base-2 " << endl; }; }; class A: public Base1, public Base2 { public: void print() { } }; int main() { A obj; obj.fun(); //调用的方法产生歧义 system("pause"); return 0; }

解决方法:使用作用域运算符::来解决,明确指向的方法,例如obj.Base1::fun()

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持自由互联。

本文共计745个文字,预计阅读时间需要3分钟。

C语言中多重继承如何避免二义性原理的实例应用有哪些?

在派生类中对基类成员访问应该是唯一的,但在多继承时可能会出现对基类成员访问不一致的情况,这就是C++多继承中的二义性问题。有两种多继承的情况会导致二义性:一是如果两个基类有同名的成员,二是如果派生类通过两个基类继承同一个成员。

在派生类中对基类成员访问应该是唯一的,但是在多继承时,可能会导致对基类某成员访问出现不一致的情况,这就是C++多继承中的二义性。

有两种继承的情况会产生多义性

一、如果一个派生类从多个基类派生,而这些基类又有一个共同的基类,则在对该基类中声明的成员变量进行访问时,可能产生二义性,继承关系如下图所示:

#include <iostream> using namespace std; class A{ public: int a; }; class B1 : public A{ public: int b1; }; class B2 : public A{ public: int b2; }; class C : public B1, public B2{ public: int c; }; int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c = 300; c1.a = 500; //报错,不能确定成员变量a具体在那个类 cout << "end..." << endl; system("pause"); return 0; }

解决方法:虚继承

C语言中多重继承如何避免二义性原理的实例应用有哪些?

注意:C++编译系统在实例化C类时,只会将虚基类A的构造函数调用一次,忽略虚基类的其他派生类(class B1,class B2)对虚继承的构造函数的调用,从而保证了虚基类的数据成员不会被多次初始化。

在虚基类A中有一个虚指针指向一个虚表,虚表中记录了虚基类与本类的地址偏移,通过这个地址偏移可以找到虚基类的成员变量a的地址

#include <iostream> using namespace std; class A{ public: int a; }; class B1 : virtual public A{ public: int b1; }; class B2 : virtual public A{ public: int b2; }; class C : public B1, public B2{ public: int c1; }; int main(){ C c1; c1.b1 = 100; c1.b2 = 200; c1.c1 = 300; c1.a = 500; //虚继承使得成员变量a只有一份拷贝,通过虚指针可以确定地址 cout << "end..." << endl; system("pause"); return 0; }

二、一个派生类同时继承两个基类,这两个基类存在相同的成员函数

#include<iostream> using namespace std; class Base1 { public: void fun() { cout << "I am base-1 " << endl; }; }; class Base2 { public: void fun() { cout << "I am base-2 " << endl; }; }; class A: public Base1, public Base2 { public: void print() { } }; int main() { A obj; obj.fun(); //调用的方法产生歧义 system("pause"); return 0; }

解决方法:使用作用域运算符::来解决,明确指向的方法,例如obj.Base1::fun()

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持自由互联。