当前位置: 首页 > news >正文

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++,因为它们不仅分配和释放内存,还能处理对象的构造和析构。

http://www.lryc.cn/news/383002.html

相关文章:

  • 如何有效地优化 Erlang 程序的内存使用,以应对大规模数据处理的需求?
  • vue3项目使用@antv/g6实现可视化流程功能
  • 【Linux网络(一)初识计算机网络】
  • Vulhub——Log4j、solr
  • linux 设置程序自启动
  • PostgreSQL 分区表与并行查询(十)
  • React Hooks使用规则:为什么不在条件语句和循环中使用它们
  • 【Docker】Consul 和API
  • Python polars学习-07 缺失值
  • 前端面试题(八)答案版
  • 在交易中出场比入场更为重要
  • 【D3.js in Action 3 精译】关于本书
  • 【408考点之数据结构】二叉树的概念与实现
  • STM32之二:时钟树
  • 第十四站:Java玫瑰金——移动开发(第二篇)
  • 数据处理技术影响皮质-皮质间诱发电位的量化
  • ResultSet的作用和类型
  • 计算机网络:运输层 - TCP首部格式 连接的创建与释放
  • 妈耶!被夸爆的零售数据分析方案在这里
  • AI探索:最佳落地应用场景
  • 2024年最新机动车签字授权人考试题库。
  • 软RAID
  • IDEA 学习之 启动“卡死”
  • 豆瓣高分项目管理书籍推荐
  • 关于docker存储overlay2相关问题
  • 实现批量自动化电商数据采集|商品详情页面|店铺商品信息|订单详情数据
  • ES6(ECMAScript 6.0) 新特性
  • 性能工具之 JMeter 常用组件介绍(八)
  • 分布式锁(Redission)
  • 【ARMv8/v9 GIC 系列 3 -- GIC 的 类型寄存器 GICD_TYPER】