C++ 二义性是什么?怎么解决?

一、什么是二义性

在多继承的场景里,当父类中存在同名变量时,子类访问父类的同名变量,将出现二义性,因为编译器不知道你将要访问的是哪个父类中的变量。

举个例子:

class A
{
public:
    int a; // B1,B2 都将继承一个变量 a
};
class B1 : public A
{
};
class B2 : public A
{
};
class C : public B1, public B2
{
};

int main()
{
    C c;
    c.a = 10; // ERROR ! 二义性 !!!
    return 0;
}

 

二、怎么解决

1. 不使用多继承

一般来说,单继承就可以满足我们 99% 的需求了,我们应该尽量避免使用多继承带来的二义性问题。(注意:这里说的单继承不包括下面说的这种类似于 “接口” 的父类)。

由于 C++ 中不存在接口,但是可以使用只包含纯虚函数的抽象类替代,如果是只包含纯虚函数的抽象类,再多继承都将不会发生二义性(父类都没有变量了当然不会有二义性)。

2. 使用虚继承

虚继承只能解决多个父类的同名变量都是从公共基类中继承而来的情况,就是下图这种:

使用虚继承:

class A
{
public:
    int a;
};
class B1 : virtual public A // 虚继承
{
};
class B2 : virtual public A // 虚继承
{
};
class C : public B1, public B2
{
};
 
int main()
{
    C c;
    c.a = 10; // OK,不会有二义性了
    return 0;
}

原理:使用虚继承时,C++ 编译器会做特殊处理,只会调用一个公共基类 A 的构造方法,这样就不会创建出多个同名变量 a 了。

3. 使用 “类名::变量名” 显性访问

还有一种二义性出现的场景,就是多个父类都是独立的,它们没有公共基类,这些独立的父类中存在同名变量的话,就不能使用虚继承来解决了,类似下图这样子:

这种情况,我们就只能使用 “类名::变量名” 显性访问,避免二义性了:

class B1
{
public:
    int a;
};
class B2
{
public:
    int a;
};
class C : public B1, public B2
{
};

int main()
{
    C c;
    c.B1::a = 10;
    c.B2::a = 20;
    return 0;
}

同样,如果父类还存在同名方法,我们也可以使用 “类名::方法名” 这样显性调用。

  • 59
    点赞
  • 162
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
多重继承会导致二义性,例如当一个派生类同时继承了两个基类,而这两个基类又继承了同一个基类,就会导致派生类中包含两个相同的基类,从而产生二义性问题。为了解决这个问题,C++ 提供了虚继承和虚基类的机制。 虚继承是一种特殊的继承方式,它可以解决多重继承时的二义性问题。在虚继承中,被继承的基类被声明为虚基类,这样就可以保证在派生类中只有一份虚基类的实例。在定义虚基类时,可以使用关键字 `virtual` 来声明,例如: ```c++ class A { public: int a; }; class B: virtual public A { // 虚继承 public: int b; }; class C: virtual public A { // 虚继承 public: int c; }; class D: public B, public C { // 多重继承 public: int d; }; ``` 在上面的例子中,`B` 和 `C` 都虚继承了 `A`,而 `D` 则同时继承了 `B` 和 `C`,这样就避免了 `D` 中包含两个相同的 `A` 的实例。 虚继承的实现方式是通过在派生类对象中添加一个指向虚基类对象的指针来实现的,虚基类的构造函数和析构函数只会被调用一次。在虚继承中,需要注意以下几点: 1. 虚基类的构造函数和析构函数不受派生类的控制,由直接或间接派生虚基类的最终派生类负责调用; 2. 在最终派生类的构造函数中,必须显式调用虚基类的构造函数,以便初始化虚基类的成员; 3. 在最终派生类的析构函数中,不需要显式调用虚基类的析构函数,系统会自动调用。 虚继承可以解决多重继承的二义性问题,但是会增加程序的复杂度,应该谨慎使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值