概念:解决菱形继承的数据冗余和二义性
原理:将虚基类对现象放到公共位置(vs是放到整个对象尾部),虚基表中存偏移量,来计算虚基类的位置;
没加virtual之前

加了virtual只后的内存情况


虚继承中,将A通常叫做虚基类

概念:虚函数重写是多态条件之一
多态原理:虚函数地址放到对象的虚表(虚函数表)中,多态指向谁调用本质是运行到对象虚表找到要调用的虚函数
析构函数可以是虚函数吗?什么场景下析构函数是虚函数?
答:可以,并且最好把基类的析构函数定义成虚函数

总结:虚函数虚基类两个地方都用到virtual关键字,它们之间没有一点关联,不要联系到一起。
满足重写,必须接口一致,并且是虚函数继承,重写虚函数的继承是一种接口继承,什么是接口继承呢?
以下程序输出结果是() :
class A
{
public:
virtual void func(int val = 1){ std::cout<<"A->"<< val <func()
}
};
class B : public A
{
public:
void func(int val=0){ std::cout<<"B->"<< val <test();//p->test(p)
return 0;
}
输出:B->1;重写,只会重写函数的实现,不对(int val=0)中的val=0干扰,可以是
void func(int a)或void fun(int)都会构成重写,所以val的值还是对象A的val
总结:子类的缺省参数不起作用,重写虚函数是一种接口继承,也就可以理解成函数、返回值、参数(包括缺省参数)都是父类继承下来,所以这里用的 父类的缺省参数
( 1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无 virtual
关键字,基类的函数将被隐藏(注意别与重载混淆)。
( 2)如果派生类的函数与基类的函数同名, 并且参数也相同, 但是基类函数没有 virtual
关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。
示例程序中:
#include
class Base
{
public:
virtual void f(float x){ cout << "Base::f(float) " << x << endl; }
void g(float x){ cout << "Base::g(float) " << x << endl; }
void h(float x){ cout << "Base::h(float) " << x << endl; }
};
class Derived : public Base
{
public:
virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }
void g(int x){ cout << "Derived::g(int) " << x << endl; }
void h(float x){ cout << "Derived::h(float) " << x << endl; }
};
( 1)函数 Derived::f(float)覆盖了 Base::f(float)。
( 2)函数 Derived::g(int)隐藏了 Base::g(float),而不是重载。
( 3)函数 Derived::h(float)隐藏了 Base::h(float),而不是覆盖。
1、虚函数的重写 2、父类的指针或引用调用才符合多态条件
多态是在运行时到指向对象的虚表中查找要调用的虚函数的地址来进行调用,如果没加*或&,就会在编译时直接确认类型调用函数的地址
//virtual void fun(){}//虚函数
virtual void fun() = 0;//纯虚函数
包含纯虚函数的类叫抽象类,抽象类不能实例化出对象

纯虚函数的作用:
1、强制子类去完成重写:父类的纯虚函数virtual void fun() = 0;子类重写:virtual void func(){}
2、表示抽象的类型,抽象就是在现实中没有对应的实体的。

在32位平台下,没有加virtual大小为1,加了之后就会变4,这是因为加了虚函数关键字,底层会有个vftptr(vitual function table pointer)虚函数表指针;
1、虚函数存在哪?
答:代码段
2、虚函数表(虚表)存在哪?
代码段(常量区)

下面两种方式等同
//mp.insert(pair(6, 6));
mp.insert(make_pair(5, 5));
日常大家喜欢用make_pair因为它不用声明模板参数,自动推

#include
using namespace std;
#include
#include 
总结:
map
1、增 insert+operator[]
2、删 erase
3、查 find+(不建议operator[])
4、改 operator[]
5、遍历 iterator+范围for -》遍历出来的数据是按k排序的,应为底层是搜索数,走的是
要注意的是map中存的是pair
键值对
C++的缺陷有哪些?
多继承就是一个问题-》菱形继承-?虚继承-》底层结构的对象模型非常复杂,且有一定效率损失
什么是菱形继承?菱形继承问题是什么?-》代码冗余,会有二义性-》如何解决?-》虚继承-》解决原理是什么?