杂项
- 在c++中,使用子类指针表示父类,不强转是编译不过的。
- 构造和析构中不要调用虚函数,因为如果产生了多态,构造调到子类的方法,子类还没有初始化;析构调了子类虚函数,子类已被析构,访问到被释放的资源,会产生隐患。
- 同样是上面一点,编译器为了保险起见,不会在构造析构中形成间接调用。
- 在构造函数中,在填虚表动作的call多半是父类在调构造,然而也有可能人造出相似的情况,就是在初始化列表里调函数,例如: :m_init(printf(".....")),这样的代码是可以编译通过的。并且也会在填虚表前执行。
判断继承的层数
继承的层数要看虚表的位置被覆盖了多少回,因为构造时会从父类一层一层往下覆盖虚表,析构反之。
判断继承和成员对象的方法
观察:
- 虚表的个数
- 填写虚表的时机
总结各种情况
父类没有虚表,自身没有虚表,成员没有虚表
都没有虚表的情况下可以直接按结构体还原
父类没有虚表,子类有虚表,成员有虚表
对象的内存结构有两个虚表分别是父类和成员的,且都在子类虚表填写动作之前完成,复写1次的是成员类,复写多次的是父类
父类有虚表,自身有虚表,成员没有
内存结构中存在一个虚表,构造完父类后,移动指针构造虚表,若成员对象有构造,则构造时机在子类填写虚表动作之前,及时内联也在之前(注意流水线优化)
父类没有虚表,子类有虚表,成员有
首4字节留给子类的虚表指针,视成员对象在类中偏移而决定成员虚表指针位置,其特点是成员对象指针在子类填写虚表前填写,成员对象构造完毕后才填写子类虚表指针,特点是子类指针只构造一次,没有复写动作,内存局部结构为:
运算符重载
在反汇编中识别为调用函数
模板
在反汇编中同样表现为函数调用,且是多处函数实现
纯虚函数
低版本中会出现19h错误
高版本会出现 DecodePoint 标志代码