1. malloc与new的区别
malloc是标准库函数,支持覆盖。new是运算符,支持重载
malloc和free仅仅分配和回收空间,而new和delete除了分配回收空间,还会调用构造函数和析构函数
malloc返回的是void类型指针,必须进行类型转换。new返回的是具体类型指针
(1)malloc和new都是在堆上开辟内存的
malloc只负责开辟内存,没有初始化功能,需要用户自己初始化;new不但开辟内存,还可以进行初始化,如new int(10);表示在堆上开辟了一个4字节的int整形内存,初始值是10,再如new int[10] ();表示在堆上开辟了一个包含10个整形元素的数组,初始值都为0。
(2)malloc是函数,开辟内存需要传入字节数,如malloc(100);表示在堆上开辟了100个字节的内存,返回void*,表示分配的堆内存的起始地址,因此malloc的返回值需要强转成指定类型的地址;new是运算符,开辟内存需要指定类型,返回指定类型的地址,因此不需要进行强转。
如堆上开辟int整形:
int *p1 = (int*)malloc(sizeof(int));
=> 根据传入字节数开辟内存,没有初始化
int *p2 = new int(0);
=> 根据指定类型int开辟一个整形内存,初始化为0
int *p3 = (int*)malloc(sizeof(int)*100);
=> 开辟400个字节的内存,相当于包含100个整形元素的数组,没有初始化
int *p4 = new int[100]();
=> 开辟400个字节的内存,100个元素的整形数组,元素都初始化为0
(3)malloc开辟内存失败返回NULL,new开辟内存失败抛出bad_alloc类型的异常,需要捕获异常才能判断内存开辟成功或失败,new运算符其实是operator new函数的调用,它底层调用的也是malloc来开辟内存的,new它比malloc多的就是初始化功能,对于类类型来说,所谓初始化,就是调用相应的构造函数。
(4)malloc开辟的内存永远是通过free来释放的;而new单个元素内存,用的是delete,如果new[]数组,用的是delete[]来释放内存的。
2.malloc与alloca的区别
alloca是向栈申请内存,因此无需释放. malloc分配的内存是位于堆中的,并且没有初始化内存的内容,因此基本上malloc之后,调用函数memset来初始化这部分的内存空间. calloc则将初始化这部分的内存,设置为0. 而realloc则对malloc申请的内存进行大小的调整.申请的内存最终需要通过函数free来释放. 而sbrk则是增加数据段的大小;
3. malloc的操作步骤
- 对堆进行加锁
- 在正式申请空间之前,对堆进行校验
- 检测申请内存块的类型
- 检测内存空间是否充足,不够设置错误信息,返回NULL,否则进行5
- 检测块的类型
- 计算本次所要申请的内存块的总字节数
- 按照计算的总字节数申请内存,底层真正向堆申请空间的是HeapAlloc函数
- 检测是否申请成功,如果申请失败设置错误信息,返回NULL,否则执行9
- 修改请求次数和目前申请的总字节数
- 将新申请的内存块的新节点头插到双向链表中
- 给该结点对应的结构体赋值
- 填充空间
- 获取申请内存块中存放有效数据的真正位置
- 对堆进行解锁
- 返回有效数据区域的地址