进程在创建的时候先是创建虚拟地址空间,并不是真正的一块空间,而是创建了一个数据结构(《程序员的自我修养》第6章 进程的创建)
继承和多态:
1)继承就是对代码的复用
2)由于构造和析构只在自身所在的类的作用域下起作用,因此基类的构造函数和析构不能被继承
3) 继承方式有三种: 公有继承(public)保护继承(protect)私有继承(private)
当继承方式为公有继承时,基类的public protect 在派生类中依旧是public protect
当继承方式为保护继承时,基类的public protect 在派生类中均为 protect
当继承方式为私有继承时,基类的public protect 在派生类中均为private
无论是哪种访问方式,基类的私有成员或成员方法均可以被继承,但在派生中不可见,不能访问
那如果我要想在派生中访问基类的私有成员变量的话,那么我可以在派生中调用能访问该成员变量或成员方法的成员方法,从而间接访问私有的数据成员;另一种不太好的办法就是将派生声明为基类的友元。
4)当继承方式缺省时,也就是省略了继承方式时,那么继承方式默认就是私有的继承方式
5)基类的成员方法不占用派生类的内存,只有成员变量占用派生类的内存。
6)派生类在继承基类的所有的数据成员和成员函数的同时,也继承了基类的作用域,从而当基类的成员变量和派生类的成员变量同名时,不会产生歧义,因为其作用域不同。
7)当出现嵌套的继承的时候,在继承时,我们只看直接的基类,而不看间接的基类。
8)继承的构造顺序:
1)构造基类中的成员对象
2)构造基类
3)构造派生中的成员对象
4)构造派生
继承的析构顺序和构造顺序相反
9)在主函数中进行函数调用时,派生类中如果没有该成员方法,则调用派生从基类继承的方法
如果不想直接调用派生的方法,想直接调用基类,则可以在函数名前加上基类的作用域
如果要构成重载函数,则必须这两个函数在同一个作用域,否则不能构成重载。
10)派生和基类中同名的成员函数之间的关系有三种:
1.重载 2.隐藏 3.覆盖
a)重载是指在同一个作用域下的函数,一般情况下,给派生或基类中的函数加上类作用域才会构成重载(作用域相同,this指针不同)。
b)隐藏:只是函数名相同,参数列表和作用域不相同,成为隐藏,派生的方法隐藏基类的同名方法。
c)覆盖:不同作用域下,函数名相同,参数列表和返回值都相同,而且基类的方法是虚函数(virtual),指针指向谁,就调用谁的成员函数。
11)动多态:运行时的多态(虚函数)
静多态:编译时的多态(重载函数,模板等,编译时就确定了要用什么类型实例化,确定要调用哪个函数)
用基类指针调用同名方法时,是否可以调用要看该基类中有无该方法(且继承方式为public),编译时看基类,运行时看派生类。
何种情况下可以将一个函数写为虚函数:
当基类指针指向堆上动态开辟的派生类,在delete基类型指针时,只会析构基类,而不能析构派生类对象时,我们可将基类的析构函数写成虚函数,这时派生类的析构也相当于是一个析构函数,从而派生类的析构可以覆盖基类的析构,继而派生类对象也被析构。
12)那么是否是所有的函数都可以作为虚函数呢?
13)当调用同名覆盖方法时是一种多态。
RTTI指的是生成对象的类型
基类对象和派生对象之间的赋值:
14)多态性:一个名字可以具有多种语义,函数重载,类模板,和函数模板都具有多态性利用多态性,用户只需要发送一般形式的消息,将所有的实现都留给接收消息的对象,对象根据接收到的消息而做出相应的操作,虚函数是实现多态性的另一重要机制。
16)编译阶段 push call函数地址
运行阶段,多态:
Mov ecx ptr dwrd[p];//将虚函数表的地址给ecx
Mov eax ptr dwrd[ecx];//将虚函数表第0行虚函数地址放入eax
Call eax;
17)析构函数为什么可以写成虚函数?
当基类指针指向堆上动态开辟的空间存放的是派生类对象时,在析构(delete 基类指针 )时,派生类对象不能被析构,因此我们需要给基类的析构函数加上virtual 使其变为虚函数,与此对应派生类中的析构也是虚函数,因此派生类析构会覆盖基类的析构。从而派生类中的析构会正常调用。
18)发生多态的三个必要条件:(缺一不可)
a)通过指针或引用调用函数
b)调用的函数是虚函数
c)对象存在
通过对象本身调用虚函数不是多态。
构造函数中调用虚函数不会产生多态,因为对象还未存在。
类型的强制转化只是为了编译通过,但这种强转是有风险的,但仍旧是构成多态,派生类指针指向基类对象,运行时类型是Base*