//继承中的构造与析构函数
//构造函数调用顺序是 基类>嵌套类>自己,析构相反
#include
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
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++中一个类可以继承多个类;并且拥有他们的属性与方法;
但有时候会产生二义性
ABCD四个类继承如图所示:
A //A是BC的基类
/ \ //BC继承A后可能分别会以不同的方式初始化A
【假设A没有无参构造函数】必须由B或者C进行初始化
B C
\ /
D //D在同时继承BC,如果初始化D的对象时,
调用B还是C的构造函数去初始化A?
这就使编译器十分矛盾;产生了二义性;
解决方法;利用虚继承:virtual关键字在继承时进行限定;
【讲多态的时候会着重讲virtual关键字】
具体实现方法如下:
#include
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关键字处理后,类的大小有所改变;具体原因在多态中
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Python工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Python开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Python开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024c (备注Python)
nvert/252731a671c1fb70aad5355a2c5eeff0.png)
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Python开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024c (备注Python)
[外链图片转存中…(img-6sFIwA8n-1711927632309)]