编译时的多态性称为静态联编. 当调用重载函数时, 在编译期就肯定下来调用哪一个函数.bash
运行时的多态性称为动态联编. 在运行时才能肯定调用哪一个函数, 由虚函数来支持.函数
派生一个类的缘由并不是老是为了添加新的成员或成员函数, 有时是为了从新定义基类的成员函数.学习
#define PI 3.14159
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
double area() {
return 0;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
double area() {
return PI * radius * radius;
}
};
int main() {
Point a(1.5, 6.7);
Circle c(1.5, 6.7, 2.5);
cout << a.area() << endl; // 调用对象a的成员函数area()
cout << c.area() << endl; // 调用对象c的成员函数area()
Point * p = &c; // 对象c的地址为指向Point类型指针赋值
cout << p -> area() << endl; // 调用Point类的成员函数area()
Point & rc = c; // 对象c初始化Point类型的引用
cout << rc.area() << endl; // 调用Point类的成员函数area()
}复制代码
在派生类有同名函数的状况下ui
Point * pPoint; // 声明的基类指针只能指向基类
复制代码
Circle * pCircle // 声明的派生类指针只能指向派生类复制代码
若是派生类没有基类的同名函数, 派生类的指针才根据继承原则调用基类的函数spa
一旦定义了虚函数, 该基类的派生类中的同名函数也自动成为虚函数.指针
用关键字virtual来声明一个虚函数, 虚函数只能是类中的一个成员函数, 不能是静态成员.code
像这样cdn
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() {
return 0;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
// 基类中area()函数为虚函数, 派生类中的同名函数也自动成为虚函数
double area() {
return PI * radius * radius;
}
};
复制代码
关键字virtual告诉编译器调用虚函数进行动态联编. 对象
使用虚函数不必定产生多态性, 也不必定使用动态联编. blog
在调用中对虚函数使用成员名限定, 能够强制编译器对该函数使用静态联编.
产生运行多态性, 也就是动态联编有3个前提
像这样
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() {
return 0;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
// 基类中area()函数为虚函数, 派生类中的同名函数也自动成为虚函数
double area() {
return PI * radius * radius;
}
};
void display(Point * p) {
cout << p -> area() << endl;
}
void display(Point & a) {
cout << a.area() << endl;
}
int main() {
Point a(1.5, 6.7);
Circle c(1.5, 6.7, 2.5);
Point * p = &c; // 对象c的地址为指向Point类型指针赋值
Point & rc = c; // 对象c初始化Point类型的引用
display(a); // 调用对象a的成员函数area()
display(p); // 根据运行时的多态性, p指向的c对象, 因此实际调用c对象的成员函数area()
display(rc); // 根据运行时的多态性, rc是对c对象的引用, 因此实际调用c对象的成员函数area()
/** 输出结果
0
19.6349
19.6349
*/
}
复制代码
在基类中不给虚函数一个有意义的定义, 能够说明为纯虚函数, 将定义留给派生类去作
像这样
class 类名 {
public:
virtual 函数类型 函数名(参数列表) = 0;
};
复制代码
抽象类: 包含有纯虚函数的类称为抽象类. 一个抽象类至少有一个纯虚函数, 一个抽象类只能做为基类来派生新类, 不能说明抽象类的对象.
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() = 0;
};
int main() {
// Point a(1.5, 6.7); // Point为抽象类, 不能实例化一个对象 error Variable type 'Point' is an abstract class
Point * p; // Point为抽象类, 能够声明一个指向抽象类对象的指针
}复制代码
注意
空虚函数定义 virtual void area() {}
纯虚函数定义 virtual void area() = 0;
纯虚函数的派生类还是抽象类. 若是派生类中给出了基类全部纯虚函数的实现, 则该派生类再也不是抽象类
若是经过同一个基类派生一系列的类, 则将这些类总称为类族.
像这样
#define PI 3.14159
// 抽象类 有一个纯虚函数 area()
class Shape {
public:
virtual double area() = 0;
};
// 正方形 有一个连长数据成员
class Square: public Shape {
protected:
double h;
public:
Square(double a) {
h = a;
}
double area() {
return h * h;
}
};
// 圆
class Circle: public Square {
public:
Circle(double a):Square(a) {
}
double area() {
return h * h * PI;
}
};
// 三角形
class Triangle: public Square {
protected:
double w;
public:
Triangle(double a, double b):Square(a) {
w = b;
}
double area() {
return w * h * 0.5;
}
};
// 矩形
class Rect: public Triangle {
public:
Rect(double a, double b):Triangle(a, b) {
}
double area() {
return w * h;
}
};
int main() {
Shape * s[5];
s[0] = new Square(4);
s[1] = new Circle(10);
s[2] = new Rect(3, 6);
s[3] = new Triangle(3, 6);
s[4] = new Square(6);
for (int i = 0; i < 5; i++) {
// 由于虚函数支持动态联编, 因此在运行时才肯定每一个元素的类型, 调用各自的成员函数
cout << "s[" << i << "] = " << s[i] -> area() << endl;
}
}
复制代码
多重继承能够被视为多个单一继承的组合.
// 基类
class A {
public:
virtual void func() {
cout << "call A" << endl;
}
};
// 基类
class B {
public:
virtual void func() {
cout << "call B" << endl;
}
};
// 多重继承
class C: public A, public B {
public:
void func() {
cout << "call C" << endl;
}
};
int main() {
A * pa;
B * pb;
C * pc, c;
pa = &c;
pb = &c;
pc = &c;
pa -> func(); // 动态联编, pa指向派生类对象c, 调用对象c的成员函数C::func();
pb -> func(); // 动态联编, pb指向派生类对象c, 调用对象c的成员函数C::func();
pc -> func(); // pc既是指向C类对象的指针, 又是指向的C类对象, 调用对象c的成员函数C::func();
}复制代码
C++有两种多态性, 一种是编译时多态性, 也叫静态联编; 另外一种是运行时多态性, 也叫动态联编. 这大大提升了咱们解决问题的丰富性. 可能也是C++长久不衰的魅力所在吧! 我会继续深刻学习C++, 继续挖掘语言的本质!