malloc、free和new、delete的区别
malloc、free是c语言中的 库函数,new、delete是c++中的 操作符。都是用来分配内存和释放内存的。
malloc | new |
---|---|
+(sizeof) (手动计算内存大小) | 类型名 (自动计算内存大小) |
返回的是void*之后进行类型转换 | 返回的是对象类型的指针 |
分配失败返回的是NULL | 分配失败抛出异常 |
虚拟内存(堆) | 物理内存(初始化成员变量) 自由存储区 |
free | delete |
---|---|
void*类型的指针 | 对象类型的指针 |
malloc多申请了16个字节,然后才是首地址。(16字节主要作用是存储内存块的主要信息)
虚函数表和虚函数表指针
实现 多态。
什么时候生成?
编译器编译时生成,发现virtual关键字修饰的函数。
存放在哪里?
- 可执行程序(磁盘)
.bss:未初始化的或者初始化为0的全局、静态变量。
.data:初始化的全局、静态变量
.rodata(readonly):虚函数表(虚函数地址的数组,指向代码的函数具体位置) - 运行状态(内存)
.bss和.data加载到静态存储区。
.rodata加载到代码区。
内核空间、栈区、文件映射区、堆区、数据区(静态存储区)、代码区
避免浅拷贝(位), 深拷贝(一个指针对应一个)。
虚函数表指针的创建时机
- 类对象构造的时候,把类的虚函数表地址赋值给vptr。
- 没有构造函数,编译器会生成默认的构造函数。
继承情况下,如果子类没有,会继续用父类的虚函数地址;如果子类有重写,会重写自己的地址。
继承情况下,虚函数表指针赋值过程
- 调用基类构造函数,把A的虚函数表的地址赋值给vptr。
- 调用子类构造函数,把B的虚函数表指针地址赋值给vptr。
class A{
public:
virtual void func();
};
class B : public class A{
public:
virtual void func();
};