C++基础语法----类的继承

简介: C++基础语法----类的继承

前言


前面介绍到了类的封装,想必大家对类的封装都有了很好的理解,今天简单说一下类的继承。继承的基本概念以及如何进行继承。



一、子类的与基类的关系:


  • 子类可以看成特殊的基类;---------------【比基类多一些新的属性或方法】
  • 子类拥有父类的成员属性与函数-------【基类公有的、被保护的可以直接用,私有的以函数为跳板可以间接使用】
  • 子类的对象可以当作基类对象使用----【与前两个性质无异】
  • 子类可以有父类没有的方法与属性----【可以定义新的属性与方法扩展】


二、继承的方式与权限:


1.基类属性子类使用权限


继承父类的方法与属性,如果在父类中是私有属性,那么无论何种属性继承在子类中都不能使用;
打个比方;
父亲的公有属性就是父亲的姓名; public            【儿子可以知道】
父亲的被保护成员就是家里面的银行密码; protected     【儿子可以共享】
父亲的私有成员就是父亲的爱人; private       【儿子不可以与父亲共享】
                        【有些秘密不可以让儿子知道】


2.继承权限


公有继承: 将基类的属性与方法继承下来;权限不变;
(最常使用):只不过基类的私有权限可以继承但是不能使用;
保护继承: 对与基类中被保护的和公有的属性,方法全部改为被保护;私有的仍为私有;
私有继承: 将父类中的属性与方法以私有权限继承下来;但父类的私有权限仍只能在父类中使用; 
         子类继承下来的成员方法,只能在子类中使用;


三、static关键字在基类时,如果属性是公有,那么也可以被子类进行共享:


需要注意的是:类中的static变量要进行显示的初始化;
初始化语法是: 类型 static 类名::变量名=XXX;(其实是在告诉编译器给他分派内存)


四、继承的构造与析构:


1 构造函数调用顺序


基类>组合类>自己;(如果级别相同那么顺序由定义顺序决定)
析构与构造相反;
• 1
• 2


2 如果基类没有不含参的构造函数,那么必须在构造子类对象时对其初始化


   语法(子类的构造函数对父类属性进行初始化):  
   子类名 (参数列表) :基类1名(参数列表),基类2名)(参数列表){函数体};
  ⚠️:在继承中子类仅仅将父类的属性与方法继承了下来,千万不要认为
     子类的对象上面还有一个父类;
  ⚠️:子类对象与父类对象的关系:(符号代表属性与变量)
  ***  爷爷类//代代相传代代变强变多【只有三颗*】
  ***@@@  爸爸类         【比爷爷类多出三个@】
  ***@@@$$$ 儿子类       【儿子类有更多的属性与方法】


3 子类初始化基类属性的方法


两个b是同一个b,可以直接初始化父类的属性
只有在基类没有无参构造函数的时候这么写
可以多传几个参数,看个人需求。参数要在子类的构造函数参数列表内。

cecfb9a9c08b4bd5828602f6efff6f04.png


4 示例代码


//继承中的构造与析构函数
//构造函数调用顺序是  基类>嵌套类>自己,析构相反
#include<iostream>
using namespace std;
class A {
private:
  int a;
public:
  A(int a ) {
    this->a = a;
    cout << "这里是A的构造函数" << endl;
  }
  void print() {
    cout << "A: "<<a << endl;
  }
  ~A() {
    cout << "这里是A的析构函数" << endl;
  }
};
class B :public A{
private:
  int b;
public:
  B(int b=0) :A(b){//要在B构造函数的初始化列表对A进行初始化,
    this->b = b;
    cout << "这里是B的构造函数" << endl;
  }
  ~B() {
    cout << "这里是B的析构函数" << endl;
  }
};
int main_01() {
  B b1(2);
  b1.print();//b1对象既有类A的属性,又有类B的属性;输出的是B类的print
  b1.A::print();//输出类A的print;
  return 0;
}


五、赋值兼容原则:


子类可以初始化父类:
   初始化时调用父类的拷贝构造函数;
   父类的指针与引用可以指向子类;
   子类有父类没有的方法,并且可以完成父类的工作;
值得注意的是:
  在基类的指针或者引用与子类的对象关联时就好像指针只与子类对象
  的一部分进行了关联;仅仅是用子类对象的一部分属性
  初始化出来了基类的对象,并且该对象中没有子类的成员属性;


具体实现方法如下:


//基类的指针与引用可以指向子类的对象;
//子类的对象可以对父类进行初始化;
#include<iostream>
using namespace std;
class A1 {
public:
  int a;
  int b;
public:
  void print() {
    cout << "A1 A1:"<<" : "<<a<<b << endl;
  }
};
class B1:public A1 {
public:
  int a;
  int b;
public:
  B1(int a, int b) {
    this->a = a;
    this->b = b;
    A1::a = 100;
    A1::b = 200;
  }
  void print() {
    cout << "    " << B1::a<<" "<< B1::b << endl;
    cout << "B1 B1"<<" : "<<a<<b << endl;
  }
  int getA() {
    return  a;
  }
};
int main_02() {
  B1 b1(1, 2);
  A1 a1= b1;//(兼容赋值原则)(子类直接给父类的对象赋值)
  A1* p = NULL;
  p = &b1;//将子类的地址赋给基类的指针;
  A1& q = b1;//基类还可以做子类的引用;
//  B1* q = NULL;
//  q = &a1;也不可以进型子类的指针指向基类的对象
//  B1 b2 = a1;可以将子类的对象赋值给父类的对象:不能反向操作;因为子类对象有父类没有的东西;
//************************分割线*******************************************************************
  cout << "" << endl;
  cout << q.a << endl;
  q.print();//打印的还是A类的东西;
  cout << "-----------------------------0-----" << endl;
  p->print();//输出的是A类内的元素;并且调用的是A类的打印函数;
  cout << p->a << endl;//打印A类的元素
  b1.print();//只有这句话调用了b的打印函数
  a1.print();//没有显示的对a1成员属性进行操作:但输出了100  200;如果将a1=b1去掉则输出乱码:
  //由此可知b1中的A1::a与A1::b对a1中的a,b进行了初始化;
  return 0;
}


六、成员属性或者成员方法重名问题:


子类与父类难免有些成员属性会重名;由于有作用域的限制子类继承时可以将同名的属性与方法继承下来;


只需在使用时稍加注意即可:


默认情况子类在调用重名属性时会先调用自己的属性;

语法:子类对象名.重名函数;

想要调用基类的是重名属性的方法:(使用域作用符):

语法:子类对象名.基类名::重名属性;【::是域作用符】



七、多继承:


在C++中一个类可以继承多个类;并且拥有他们的属性与方法;
但有时候会产生二义性
• 1
• 2


情况一、虚继承解决构造函数的二义性


ABCD四个类继承如图所示: 
               A      //A是BC的基类
               /    \ //BC继承A后可能分别会以不同的方式初始化A
                    【假设A没有无参构造函数】必须由B或者C进行初始化
             B       C  
               \    /
                D   //D在同时继承BC,如果初始化D的对象时,
                      调用B还是C的构造函数去初始化A?
这就使编译器十分矛盾;产生了二义性;
解决方法;利用虚继承:virtual关键字在继承时进行限定;
【讲多态的时候会着重讲virtual关键字】


具体实现方法如下:


#include<iostream>
using namespace std;
class AA {
public:
  int a;
  AA(int a = 0) {
    this->a = a;
  }
  void print() {
    cout << " AA" << endl;
  }
};
class BB:virtual public AA {//
public:
  int b;
  BB(int b = 0):AA(1) {
    this->b = b;
  }
  void print() {
    cout << "BB" << endl;
  }
};
class CC:virtual public AA {//
public:
  int c;
  CC(int c = 0):AA(2) {
    this->c = c;
  }
  void print() {
    cout << "CC" << endl;
  }
};
class DD :public BB,public CC{
public:
  int d;
  //初始化继承到的基类
  DD(int d=0):BB(d),CC(d),AA(d) {//如果不是虚继承 那么不可以直接用DD去初始化AA
    this->d = d;
  }
  void ptint() {
    cout << "DD" << endl;
  }
};
int main() {
  DD d1(666);
  cout << "b  c  d" << endl << d1.b << d1.c << d1.d << endl;
  cout << d1.a << endl;
//  cout << d1.a << endl;//不明确A是哪个里面的---------解决方案---虚继承virtual
  cout << d1.AA::a << endl;//没有虚继承时输出了BB对象的属性
  cout << d1.BB::a << endl;//虚继承后输出的均是最后一次对该成员变量的操作的值
  cout << d1.CC::a << endl;//virtual会对其自动进行改变
  cout << "BB  " << sizeof(BB) << "CC  " << sizeof(CC) << "DD  " << sizeof(DD) << endl;//虚继承输出12 12 24
  return 0;//不虚继承输出 8 8 20;
}


用virtual关键字处理后,类的大小有所改变;具体原因在多态中


情况二、基类重名变量


A         B  //C同时继承了AB,但AB中有一变量名相同,
          \    /
          C   //在C通过域作用符调用AB中的同名变量时;会产生二义性;



解决方法:


在C的对象调用那个变量时加上AB的作用符进行限定;
比如使用A中的aa属性调用方法则为
          C.A::aa
• 1
• 2
• 3


总结


这些仅仅是C++类继承中的一些浅显的知识点,希望大家能够熟练掌握,至少要知道继承的几种权限,以及继承来的属性是否可以使用,还有虚继承、多继承的几种问题。

相关文章
|
3月前
|
存储 编译器 开发工具
01.C++基础语法
本教程系统讲解C++基础语法,涵盖语言特性、编译器使用、命名空间、头文件、修饰符与关键字等核心内容,助你快速掌握C++编程基础。
292 9
|
3月前
|
安全 Java 编译器
C++进阶(1)——继承
本文系统讲解C++继承机制,涵盖继承定义、访问限定符、派生类默认成员函数、菱形虚拟继承原理及组合与继承对比,深入剖析其在代码复用与面向对象设计中的应用。
|
9月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
368 12
|
7月前
|
存储 安全 Java
c++--继承
c++作为面向对象的语言三大特点其中之一就是继承,那么继承到底有何奥妙呢?继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用,继承就是类方法的复用。
163 0
|
7月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
197 0
|
7月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
305 0
|
10月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
196 16
|
10月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
10月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
10月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
532 6