malloc、free和new delete的区别

malloc/freenew/delete 是在 C++ 中分配和释放内存的两种不同方法。它们主要有以下区别:

1. 语法和用法

  • mallocfree: malloc开辟空间时需要手动计算分配的空间大小

    int* p = (int*)malloc(sizeof(int) * 10); // 分配10个int类型的内存
    // 使用内存
    free(p); // 释放内存
    

    实际malloc在分配空间的时候会多给我们分配16个字节的空间(存储了内存块的描述信息),即16+mem。然后返回mem的首地址。然后free的时候,会以mem的地址左偏移16位,这样就知道能够释放多大的空间了。

  • newdelete: new开辟空间不需要手动计算分配的大小

    int* p = new int[10]; // 分配10个int类型的内存
    // 使用内存
    delete[] p; // 释放内存
    

2. 类型安全

  • mallocfree: malloc 返回 void*,需要显式转换为所需的类型指针,不提供类型安全。
  • newdelete: new 直接返回所需类型的指针,不需要类型转换,提供类型安全。

3. 构造函数和析构函数

  • mallocfree: 只分配和释放原始内存,不调用构造函数和析构函数。适用于 C 语言风格的内存管理。
  • newdelete: new 在分配内存后调用构造函数,delete 在释放内存前调用析构函数,适用于需要对象初始化和清理的场景。

4. 内存分配失败处理

  • malloc: 内存分配失败时返回 NULL,需要显式检查。
  • new: 内存分配失败时抛出 std::bad_alloc 异常。

5. 自定义操作符

  • newdelete: 可以重载自定义的 newdelete 操作符来实现特定的内存分配行为。

6. 适用范围

  • mallocfree: 主要用于 C 语言,也可以在 C++ 中使用,但不推荐用于需要对象初始化的场景。
  • newdelete: 专为 C++ 设计,推荐用于分配和释放 C++ 对象。

7. 内存分配地址

  • malloc:是在堆上分配的,如果分配的内存小于128k一般是在内存池中取用。如果大于128k则通常会使用 mmap 系统调用直接从操作系统请求内存。mmap 会映射一个匿名内存区域到进程的地址空间,并返回该区域的地址。这种方法的优点是大块内存可以独立管理和释放,不会影响到常规的内存池。

  • new: 是在free sotre上分配内存

    • 调用 operator new 分配内存:operator new 是一个内置的或用户自定义的函数,用于从自由存储区分配足够的内存。
      标准库提供了默认实现的 operator new,通常会调用底层的内存分配函数(如 malloc)来分配内存。
      operator new 可能会抛出 std::bad_alloc 异常,如果内存分配失败。
      调用对象的构造函数:

    • 在成功分配内存后,new 运算符会在分配的内存地址上调用对象的构造函数。
      这一步骤确保对象被正确地初始化。
      返回对象的指针:

    • 构造函数调用完成后,new 运算符返回指向新分配和构造的对象的指针。

示例对比

mallocfree 示例
#include <cstdlib> // 包含 malloc 和 free 的头文件
#include <iostream>

struct MyStruct {
    int x;
    MyStruct() : x(10) {} // 自定义构造函数
};

int main() {
    MyStruct* p = (MyStruct*)malloc(sizeof(MyStruct)); // 只分配内存,不调用构造函数
    if (p == nullptr) {
        std::cerr << "Memory allocation failed\n";
        return 1;
    }
    p->x = 20; // 需要手动初始化
    std::cout << "MyStruct.x = " << p->x << std::endl;
    free(p); // 只释放内存,不调用析构函数
    return 0;
}
newdelete 示例
#include <iostream>

struct MyStruct {
    int x;
    MyStruct() : x(10) {} // 自定义构造函数
    ~MyStruct() { std::cout << "Destructor called\n"; } // 自定义析构函数
};

int main() {
    MyStruct* p = new MyStruct; // 分配内存并调用构造函数
    std::cout << "MyStruct.x = " << p->x << std::endl;
    delete p; // 调用析构函数并释放内存
    return 0;
}

总结来说,malloc/free 主要用于 C 风格的内存管理,而 new/delete 适用于 C++,因为它们不仅分配和释放内存,还能处理对象的构造和析构。

  • 9
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: mallocfree是C语言中的动态内存分配和释放函数,用于在程序运行时动态地分配和释放内存空间。malloc函数用于分配一块指定大小的内存空间,free函数用于释放已分配的内存空间。 newdeleteC++语言中的动态内存分配和释放运算符,用于在程序运行时动态地分配和释放内存空间。new运算符用于分配一块指定大小的内存空间,并返回指向该内存空间的指针,delete运算符用于释放已分配的内存空间。 总的来说,mallocfree是C语言中的动态内存分配和释放函数,newdeleteC++语言中的动态内存分配和释放运算符。它们的使用方法和功能类似,但是在C++中,newdelete还可以用于对象的构造和析构。 ### 回答2: 1. mallocfree 它们是C语言中的动态内存分配和释放函数,用于动态分配和释放内存,即在程序运行时根据需要从系统堆中分配出一块连续的、指定大小的内存,并返回指向这块内存的指针;当程序不需要使用这块内存时,使用free函数将其释放回内存堆。由于mallocfree是C语言中的函数,因此在使用时需要手动指定内存大小,且不支持自动释放内存。 2. newdelete 它们是C++语言中的动态内存分配和释放方式,和malloc/free相比,newdelete更加方便、安全,且支持自动释放内存。new关键字可以自动计算需要的内存大小并返回对应类型的指针,delete关键字可以自动释放内存,而无需手动指定内存大小。 newdelete的内存管理是面向对象的,它们可以调用构造函数和析构函数来初始化和清除对象,可以确保正确的内存布局和类型安全,这对于C++编程来说非常重要。 3. 区别对比 从功能上来说,mallocfree是C语言中的函数,用于动态内存分配和释放,需要手动指定内存大小,不支持自动释放内存;而newdeleteC++语言中的关键字,支持自动计算内存大小、自动调用类型的构造函数和析构函数、自动释放内存。 从安全性和易用性上来说,new/deletemalloc/free更安全、更方便,由于支持自动计算内存大小和类型安全,可以避免因内存管理不当而发生错误。同时,new/delete也更加适用于面向对象的程序设计,可以自动调用构造函数和析构函数,确保对象的正确创建、初始化和清除。因此,在C++编程中,建议使用newdelete来进行动态内存管理。 总之,malloc/freenew/delete都是动态内存管理的方式,根据不同编程语言和场景需求,选择适合的动态内存分配和释放方式非常重要,同时注意内存分配和释放的规范和安全性。 ### 回答3: mallocfreenewdelete 都是用于动态内存分配和释放的关键字。它们在使用方式、机制、效率和能力上存在差异。 1. mallocfree malloc 是 C/C ++ 程序中用于动态分配内存的函数,它会根据用户的需求分配一块指定大小的内存块并返回其首地址,通常用于分配结构体或数组等动态内存。而 free 用于释放之前分配的由 malloc 分配的内存块,将内存块标记为可重用状态。 虽然 mallocfree 可以做到动态分配和释放内存,但是它们却没有处理对象的构造和析构,因此在使用过程中,如果需要先执行构造函数再分配空间或者先释放空间再执行析构函数,就需要用到以下两个关键字。 2. newdelete newdeleteC++ 中的关键字,它们不仅可以进行动态内存的分配和释放,而且还可以调用对象的构造和析构函数,能够在对象构造时自动执行构造函数,在对象销毁时自动执行析构函数,因此是一种更高级别的内存管理方式。 new 用于分配动态内存并调用构造函数来初始化对象,而 delete 用于调用析构函数并释放分配的内存,从而在对象生命周期的开头和结尾负责构造函数和析构函数的调用。 总之,mallocfree 是 C 语言的标准库关键字,而 newdeleteC++ 的关键字,它们之间的区别在于对于对象是否需要额外的构造函数和析构函数的调用等方面。因此,在实际的开发过程中,应根据需要选择合适的内存分配和释放方式。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值