C++三大特性:封装、继承、多态,今天给大家好好说说继承的奥妙
1、虚函数:
C++的虚函数主要作用是“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现。子类可以重写父类的虚函数实现子类的特殊化。
2、纯虚函数:
C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。
C++中的纯虚函数更像是“只提供申明,没有实现”,是对子类的约束,是“接口继承”。
C++中的纯虚函数也是一种“运行时多态”。
3、普通函数:
普通函数是静态编译的,没有运行时多态,只会根据指针或引用的“字面值”类对象,调用自己的普通函数。
普通函数是父类为子类提供的“强制实现”。
因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。
4、看具体运用实例:
1 #include <iostream>
2 using namespace std;
3
4 class A
5 {
6 public:
7 virtual void out1()=0; ///由子类实现
8 virtual ~A(){};
9 virtual void out2() ///默认实现
10 {
11 cout<<"A(out2)"<<endl;
12 }
13 void out3() ///强制实现
14 {
15 cout<<"A(out3)"<<endl;
16 }
17 };
18
19 class B:public A
20 {
21 public:
22 virtual ~B(){};
23 void out1()
24 {
25 cout<<"B(out1)"<<endl;
26 }
27 void out2()
28 {
29 cout<<"B(out2)"<<endl;
30 }
31 void out3()
32 {
33 cout<<"B(out3)"<<endl;
34 }
35 };
36
37 int main()
38 {
39 A *ab=new B;
40 ab->out1();
41 ab->out2();
42 ab->out3();
43 cout<<"************************"<<endl;
44 B *bb=new B;
45 bb->out1();
46 bb->out2();
47 bb->out3();
48
49 delete ab;
50 delete bb;
51 return 0;
52 }
5、总结:
因为继承最主要的场景就是有多个类用的接口基本上一样,那么就需要一个抽象类,如果这个抽象类全是纯虚函数,
那么子类需要全部继承并且定义,这样带来的问题就是可能有的类不需要这个接口但是还必须写上且空定义,这样很累,
那就需要一部分虚函数接口,所以根据实际情况,组合纯虚函数和虚函数来使用。
6、在c++中,父类和子类虚函数都有对应的虚函数表,如果子类重写了虚函数,就会将新写的这个函数的地址放到他自己的表内,并且通过父类的指针指向子类所开辟的空间,这样就可以通过父类对象调用子类的方法。
class Base
{
public:
void virtual Func()
{
cout << "Base\n";
}
};
class Derived : public Base
{
public:
void Func()
{
cout << "Derived\n";
}
};
void main()
{
Base* a = new Base;
a->Func();
输出:base
Base* b = new Derived();
b->Func();
输出:device
}