首页 > 代码库 > C++基础之三大特性之多态(顾客,别走啊,我这是羊肉,不是狗肉啊)

C++基础之三大特性之多态(顾客,别走啊,我这是羊肉,不是狗肉啊)

<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);">多态:多种形态</span>

举个例子,你有两个fun函数,第一个类中fun的功能是打印1,第二个类的fun的功能是打印2,你建一个类的对象,调fun,想打印1就能打印1,想打印2就能打印2,这就是多态,这就是多种形态,这也就是我们代码中需要达到的要求

多态的本质:灵活的复用

多态的前提:继承的基础上

多态的实现:虚函数(有的人把重载这些也当作多态,静态的多态,这种东西看个人理解,不用扣这些字眼,下面才是重中之重---虚函数)

虚函数关键字:virtual

知识点一:

class A                 void A::fun()      class B : public A<span style="white-space:pre">		</span>void B::fun()
{<span style="white-space:pre">			</span>{<span style="white-space:pre">		</span>{<span style="white-space:pre">				</span>{
public:<span style="white-space:pre">			</span>printf("1");<span style="white-space:pre">	</span>public:<span style="white-space:pre">				</span>printf("2");
virtual void fun();<span style="white-space:pre">	</span>}<span style="white-space:pre">		</span>virtual void fun();<span style="white-space:pre">		</span>}
};<span style="white-space:pre">					</span>};
B *p = new B;   p->fun();
需求:不碰最后一行代码,怎么能调到A中的fun();

我们讲一下虚函数的原理

当类中有虚函数,建类的对象,对象空间的前四个字节放的是虚表地址,通过虚表地址找到虚表,虚表中有这个类的所有虚函数的地址

当继承有虚函数的类后,例如上面的B类,B类对象的前四个字节就有字节的虚表,如果B类有自己的虚函数,B类对象前四个字节是虚表的地址,地址找到的是自己类的虚函数地址----如果B类没有自己类的虚函数,它仍有虚表(继承),虚表找到的虚函数地址是父类(即上面A类)虚函数的地址

那如果要改上面的情况,只要把B类的fun函数擦掉就可以了

知识点二:

有没有注意到我子类的fun前面也加了虚函数的关键字呢,这是个技巧

防止作用

如果有一天有一个C类继承了我现在的子类B类,那B类就成了父类,建了C类的对象,C类对象有自己的fun函数,我的要求是调用C类的fun函数,但是如果有人捣乱或者不清楚,将得到C类对象地址的指针强转成B类的指针,调到的就是B类的fun函数了

知识点三:虚析构(应付下面这种情况)

下面这道题有什么问题吗

calss A<span style="white-space:pre">				</span>class B
{<span style="white-space:pre">				</span>{<span style="white-space:pre">				</span>A *p = new B;
public:<span style="white-space:pre">				</span>public:<span style="white-space:pre">				</span>delete p;
char *p;<span style="white-space:pre">			</span>char *t;
A(){ p = new char[20];<span style="white-space:pre">	</span>}<span style="white-space:pre">	</span>B(){ t = new char[20]; }
~A(){ delete p; }<span style="white-space:pre">		</span>~B(){<span style="white-space:pre">	</span>delete t; }
};<span style="white-space:pre">				</span>};
我们看一下反汇编


这里Cs是子类,就是上面的B,上面的代码执行汇编是这样,看到01001757行代码调用子类的析构

所以这道题是B类的成员指针new出的空间泄漏了

当我们将父类的析构函数前加个virtual再看汇编

01001878调用的edx里面的内容在子类析构后最后调父类的析构

虚析构,调子类的析构---调父类析构

顺序为什么是先子后父呢---你要子类还用着,就析构了父类不就野指针了~~

虚析构是个好习惯

知识点四:纯虚函数

虚函数声明的时候后面写 = 0,这个类就是抽象类,这个函数就是纯虚函数

拥有纯虚函数的类不能建对象

如果HR问我纯虚函数,我就说 =0和不能建这个类的对象,我不会说抽象类这些概念,什么叫抽象

什么时候用到纯虚函数,就是这个类你确定以后不会建对象了,就写成纯虚函数

C++基础之三大特性之多态(顾客,别走啊,我这是羊肉,不是狗肉啊)