菱形继承和C++相关问题

简介: 菱形继承和C++相关问题

菱形继承

菱形继承是因为多继承存在问题而存在的

像这样都是a的话出现二义性,要采用virtual继承,所以菱形继承它也叫虚继承

#include<iostream>
#include<string>
using namespace std;
class A
{
public:
  A(int a) : a(a){}
protected:
  int a = 555;
};
class B : public A
{
public:
  B(int a):A(a){}
};
class C : public A
{
public:
  C(int a): A(a){}
};
class D : public B, public C
{
public:
  D(int a): B(90), C(68) {}
  void print()
  {
    cout << A::a << endl;
    cout << B::a << endl;
    cout << C::a << endl;
  }
};
int main()
{
  D d(10);
  d.print();
  system("pause");
  return 0;
}

如结果所示,存在二义性问题,这a应该都是一个值,采用virtual继承的方式来解决

在B,C的继承方式前加个virtual,这样子类就必须调用爷爷的构造函数


         

#include

#include

using namespace std;

class A

{

public:

A(int a) : a(a){}

protected:

int a = 555;

};

class B : virtual public A

{

public:

B(int a):A(a){}

};

class C : virtual public A

{

public:

C(int a): A(a){}

};

class D : public B, public C

{

public:

D(int a): B(90), C(68),A(a) {}

void print()

{

cout << A::a << endl;

cout << B::a << endl;

cout << C::a << endl;

}

};

int main()

{

D d(10);

d.print();
system("pause");
return 0;

}

如图所示,打印的结果都是10,这样就不存在二义性问题了

继承中的同名问题

就2种

1.数据成员同名

2.函数同名

在对象的访问中:

如果不加以修饰,那么按照就近原则,

如果用类名限定了,那么自然就调用那类中的

用指针访问时候

注意的是:可以用父类的指针,对子类对象进行初始化,但不能用子类的指针对父类初始化

1.如果不加以修饰,那么调用的,取决于指针的类型,如果是父类的指针类型,那么肯定就调用父类的。

#include<iostream>
#include<string>
using namespace std;
class father
{
public:
  father(){}
  father(string name):name(name){}
protected:
   string name;
};
class son : public father
{
public:
  son(){}
  son(string name1, string name2) :father("father")
  {
    this->name = "son";
  }
  void print()
  {
    cout << son :: name << endl;
    cout << father::name << endl;
  }
protected:
  string name;
};
int main()
{
  son m;
  m.print();
  father* p = new son;
  p->print();
  system("pause");
  return 0;
}

如果所示,这里是父类的指针类型,只能调用父类的东西,调用了子类的东西,编译器自然就报错了。

重载加常量情况

主要是: 有个MM类,它的对象为mm

1.mm = mm + 1

2.mm = 1 + mm

运算符重载的使用情况

#include<iostream>
#include<string>
using namespace std;
class MM
{
public:
  MM(int date) : date(date) {}
  MM operator + (int m)
  {
    this->date = this->date + m;
    return *this;
  }
  void print()
  {
    cout << date << endl;
  }
private:
  int date;
};
int main()
{
  MM mm(1);
  mm = mm + 1;  // 采用的函数运算符重载 , 实质是:  mm.成员函数()
  mm = 1 + mm; // 报错的原因是  1.成员函数()  1无法完成这种转化
  mm.print();
  system("pause");
  return 0;
}

解决这一问题的方法:就是要使用友元函数的运算符重载

#include<iostream>
#include<string>
using namespace std;
class MM
{
public:
  MM(int date) : date(date) {}
  /*MM operator + (int m)
  {
    this->date = this->date + m;
    return *this;
  }*/
  friend MM operator + (MM  date1,MM date2)
  {
    return MM(date1.date + date2.date);
  }
  void print()
  {
    cout << date << endl;
  }
private:
  int date;
};
int main()
{
  MM mm(1);
  mm = mm + 1;  // 采用的函数运算符重载 , 实质是:  mm.成员函数()
  mm = 1 + mm; // 报错的原因是  1.成员函数()  1无法完成这种转化
  //使用友元函数重载,来解决这一问题
  mm.print();
  system("pause");
  return 0;
}

委托构造

构造委托也叫做委托构造,它能够允许有一个构造函数调用另外一个构造函数

它只能采用初始化列表

c++中大部分情况,都需要初始化列表,所以建议,最好每次都是用初始化参数列表

#include<iostream>
#include<string>
using namespace std;
class MM
{
public:
  MM(): MM(10, "温柔了岁月") {}  //这就是委托构造,采用初始化列表
  MM(int age , string name) : age(age),name(name){}
private:
  int age;
  string name;
};
int main()
{
  system("pause");
  return  0;
}



相关文章
|
2月前
|
编译器 C++
【C++】详解C++的继承
【C++】详解C++的继承
|
8天前
|
C++
C++(二十)继承
本文介绍了C++中的继承特性,包括公有、保护和私有继承,并解释了虚继承的作用。通过示例展示了派生类如何从基类继承属性和方法,并保持自身的独特性。此外,还详细说明了派生类构造函数的语法格式及构造顺序,提供了具体的代码示例帮助理解。
|
29天前
|
安全 Java 编译器
|
2月前
|
存储 Java 程序员
【c++】继承深度解剖
【c++】继承深度解剖
25 1
|
2月前
|
存储 编译器 数据安全/隐私保护
|
2月前
|
Java C++ 运维
开发与运维函数问题之C++中有哪些继承方式如何解决
开发与运维函数问题之C++中有哪些继承方式如何解决
21 0
|
2月前
|
编译器 C++ 开发者
C++一分钟之-多重继承与菱形问题
【7月更文挑战第19天】C++的多重继承允许类从多个基类派生,但引入了菱形问题,即类D通过B和C(都继承自A)双重继承A,可能导致数据冗余和二义性。解决这个问题的关键是**虚继承**,通过`virtual`关键字确保基类A只被继承一次,消除冲突。理解并适当使用虚继承是处理这类问题的关键,有助于保持代码的清晰和正确性。
21 0
|
3月前
|
C++
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
|
2月前
|
存储 编译器 C++
C++基础知识(六:继承)
多态是面向对象编程的四大基本原则之一,它让程序能够以统一的接口处理不同的对象类型,从而实现了接口与实现分离,提高了代码的灵活性和复用性。多态主要体现在两个层面:静态多态(编译时多态,如函数重载)和动态多态(运行时多态,主要通过虚函数实现)。
|
3月前
|
程序员 编译器 C++
【c++】继承
【c++】继承
17 0