继承允许我们依据一个类来定义另一个类,这使得创建和维护一个应用程序变得更容易。这样做也达到了重用代码功能和提高执行效率的效果。
派生类的成员可以直接访问基类的保护成员(protected),但不能直接访问基类的私有成员(private)。不过需要注意的是,派生类的成员函数只能访问所作用的那个对象(即this指针指向的对象)的基类保护成员,不能访问其他基类对象的基类保护成员(比如通过函数参数传来的基类对象)。
继承分为公有继承、保护继承与私有继承,除了公有继承,剩下两个很少用到,三者区别如下:
C++规定在同一作用域中,函数名相同但函数特征标(即参数个数、类型、顺序)不同时,构成函数重载。
函数重载的注意事项:
int func(int a)
和 int func(int a = 0)
是不能在同一作用域中同时存在的。这里还要特别注意一下const修饰函数或函数参数时的情况:
class A {
public:
/**
* 不管形参有没有const修饰实参都不会被修改,二者在调用时没有区别,因此不能构成重载
*/
void func(int a);
void func(const int a); // NO
/**
* 由底层const修饰的指针指向的实参不能被修改,二者在调用时存在区别,因此可以构成重载
*/
void func_bot_p(int *a);
void func_bot_p(const int *a); // YES
/**
* 不管有没有顶层const修饰,该指针指向的内容都可以被修改,二者在调用时没有区别,因此不能构成重载
*/
void func_top_p(int *a);
void func_top_p(int *const a); // NO
/**
* 在const修饰引用时实参不能被修改,二者在调用时存在区别,因此可以构成重载
*/
void func_ref(int &a);
void func_ref(const int &a); // YES
/**
* 由const修饰的成员函数只能由const对象调用,二者在调用时存在区别,因此可以构成重载
*/
void func_ret(int a);
void func_ret(int a) const; // YES
};
不同作用域中定义的同名函数会构成函数隐藏(不要求函数返回值和函数参数类型相同)。
类成员函数会屏蔽全局函数,派生类成员函数会屏蔽与其同名的基类成员函数(但如果该基类成员函数为虚函数,且函数返回值和特征标相同则构成函数重写)。
#include
using namespace std;
void func() {
cout << "global::func()" << endl;
}
class A {
public:
/**
* 隐藏了外部的func
*/
void func() {
cout << "A::func()" << endl;
}
void use_func() {
func();
::func(); // 使用全局函数时要加作用域
}
};
class B : public A {
public:
/**
* 隐藏了基类的func
*/
void func() {
cout << "B::func()" << endl;
}
void use_func() {
func();
A::func(); // 使用基类函数时要加作用域
}
};
int main() {
A a;
B b;
a.use_func();
b.use_func();
}
atreus@MacBook-Pro % g++ main.cpp -o main -std=c++11
atreus@MacBook-Pro % ./main
A::func()
global::func()
B::func()
A::func()
atreus@MacBook-Pro %
派生类中与基类同返回值类型、同名和同特征标的虚函数重定义,构成虚函数覆盖,也叫虚函数重写。
需要注意的是,在默认情况下,如果重新定义了继承的方法,应确保与原来的原型完全相同,但如果返回类型是基类引用或指针,则可以修改为指向派生类的引用或指针,这种新出现的特性叫做返回类型协变(covariance of return type)。
#include
using namespace std;
class A {
public:
void func() {
cout << "A::func()" << endl;
}
virtual void func_v() {
cout << "A::func_v()" << endl;
}
};
class B : public A {
public:
/* 函数隐藏 */
void func() {
cout << "B::func()" << endl;
}
/* 函数重载 */
void func_v() override {
cout << "B::func_v()" << endl;
}
};
int main() {
A *a = new B;
a->func();
a->func_v();
delete a;
}
atreus@MacBook-Pro % g++ main.cpp -o main -std=c++11
atreus@MacBook-Pro % ./main
A::func()
B::func_v()
atreus@MacBook-Pro %
多态是指一个方法同时具有多种形态,具体形态取决于调用该方法的具体对象。从实现的角度可以将多态分为编译时多态(主要通过函数模板、函数重载和运算符重载实现)和运行时多态(主要通过虚函数和函数重写实现)。
对于运行时多态,其实现主要有三个前提:
运行时多态的实现要借助于动态绑定,动态绑定借助于虚函数实现,虚函数的限制如下:
虚函数、虚函数表及虚函数实现多态的原理
其中,动态绑定是运行时绑定,通过地址实现,它是指基类的指针或引用有可能指向不同的派生类对象。对于非虚函数,执行时实际调用该函数的对象类型即为该指针或引用的静态类型。而对于虚函数,执行时实际调用该函数的对象类型为该指针或引用所指对象的实际类型。
当类声明中包含纯虚函数(定义是末尾有 = 0
的虚函数)时,则不能创建该类的对象,这个类变为抽象类,C++中的抽象类类似于Java中的接口,抽象类必须至少包含一个纯虚函数。
此外,对于抽象类还有以下注意事项:
#include
/* 抽象类 */
class Car {
public:
virtual void showName() = 0; // 纯虚函数
};
class Audi : public Car {
public:
void showName() override { std::cout << "Audi" << std::endl; }
};
class Volvo : public Car {
public:
void showName() override { std::cout << "Volvo" << std::endl; }
};
int main() {
Audi audi;
Volvo volvo;
audi.showName(); // Audi
volvo.showName(); // Volvo
return 0;
}
菱形继承是指当类B和类C同时继承于基类A,类D同时继承于类B和类C,此时类A中的成员变量和成员函数继承到类D中就变成了两份,在D中调用A中的成员会导致二义性,同时一个变量分两份存储也存在内存空间浪费的问题。
通过虚基类和虚继承机制,可以在多继承中只保留一份共同成员,从而解决了多继承导致的命名冲突和数据冗余。
在继承方式前面加上 virtual
关键字就是虚继承,如果不采用虚继承,在类D中使用类A中的m_a时则需要通过 B::m_a
或 C::m_a
来指定具体使用哪个m_a。
#include
using namespace std;
class A {
protected:
int m_a = 0;
};
class B : virtual public A {
protected:
int m_b = 1;
};
class C : virtual public A {
protected:
int m_c = 2;
};
class D : public B, public C {
protected:
int m_d = 3;
public:
D() {
cout << m_a << endl;
cout << m_b << endl;
cout << m_c << endl;
cout << m_d << endl;
}
};
int main() {
D d;
return 0;
}
atreus@MacBook-Pro % g++ main.cpp -o main -std=c++11
atreus@MacBook-Pro % ./main
0
1
2
3
atreus@MacBook-Pro %
C++标准库中的iostream类就是一个虚继承的实际应用案例。iostream从istream和ostream直接继承而来,而istream和ostream又都继承自一个共同的名为base_ios的类,是典型的菱形继承。
参考:
https://cloud.tencent.com/developer/article/1177174
https://blog.csdn.net/weixin_39640298/article/details/88725073
http://c.biancheng.net/view/2280.html