C++中new/delete与malloc/free的区别?(构造/析构函数的调用)

发布时间 - 2026-01-09 00:00:00    点击率:
new/delete会调用构造/析构函数,malloc/free不会;前者是类型感知的操作符,后者是纯内存操作;混用导致未定义行为,且无法跨语言或跨分配器安全使用。

new/delete 会调用构造/析构函数,malloc/free 不会

这是最核心的区别。当你用 new 分配对象时,它不仅申请内存,还会自动调用该类型的构造函数;delete 在释放前会先调用析构函数。而 mallocfree 是纯内存操作,完全不感知类型,更不会触碰任何构造或析构逻辑。

常见错误现象:
malloc 分配一个 std::string 对象的内存,然后直接赋值——此时 std::string 的内部指针未被初始化,后续访问必然崩溃;或者用 free 释放 new 出来的对象,析构函数不执行,资源泄漏(比如文件句柄、动态分配的缓冲区)。

  • new MyClass → 先 operator new 分配内存,再调用 MyClass::MyClass()
  • delete ptr → 先调用 MyClass::~MyClass(),再调用 operator delete
  • malloc(sizeof(MyClass)) → 只返回一块原始字节,MyClass 的成员(尤其是非 POD 类型)处于未定义状态
  • free(ptr) → 仅归还内存,不做任何类型相关清理

不能混用:new 配 delete,malloc 配 free

混用会导致未定义行为(UB),而且往往在特定平台或优化级别下才暴露问题,非常难排查。

典型场景:
— 在 C 接口(如 dlopen 加载的库)中拿到 malloc 出来的内存,却用 delete 释放
— 把 new char[1024] 的指针传给只认 void* 的 C 函数,之后用 free 释放

  • new / new[] 必须配 delete / delete[](否则数组析构可能只调第一个元素)
  • malloc / calloc / realloc 必须配 free
  • 即使 MyClass 是 POD 类型,也不能用 malloc+delete:因为 delete 内部仍会尝试调用析构函数(哪怕为空),而该内存并非 new 分配,operator delete 可能崩溃

new/delete 可被重载,malloc/free 是固定符号

newdelete 是可被类或全局重载的操作符,你可以控制内存分配策略(比如对象池、对齐要求、日志记录);而 mallocfree 是 C 标准库函数,符号固定,无法按类型定制。

使用场景:
— 自定义 allocator(如游戏引擎中为 GameObject 专用堆)
— 调试内存泄漏时,在全局 operator new 中记录调用栈
— 硬实时系统中避免 malloc 的不可预测延迟

  • 重载 operator new 后,new MyClass 就走你的实现,但 malloc(sizeof(MyClass)) 依然走 libc 的 malloc
  • malloc 返回的指针不能直接用于 placement new 构造,除非你确保内存对齐满足类型要求(例如 alignas(std::max_align_t)
  • std::allocator 底层通常基于 operator new,而非 malloc(尽管某些旧实现可能 fallback)
class Widget {
public:
    Widget() { std::cout << "ctor\n"; }
    ~Widget() { std::cout << "dtor\n"; }
};

// 正确:构造/析构完整 Widget* w1 = new Widget; delete w1; // 输出 ctor → dtor

// 危险:未构造就用,未析构就丢 Widget w2 = static_cast>(malloc(sizeof(Widget))); new(w2) Widget; // placement new 手动构造 w2->~Widget(); // 必须手动析构 free(w2); // 不能用 delete

C++ 对象生命周期必须由构造/析构函数管理,绕过它们就像绕过交通灯——一时省事,后患是 crash 或静默损坏。尤其在有虚函数、成员含智能指针或容器的类里,跳过 new/delete 几乎必然出错。


# 字节  #   # c++  # nas  # 区别  # 标准库  # String  # 构造函数  # 析构函数  # char  # void  # 指针  # 虚函数  # 接口  #   # operator  # delete  # 对象  # 这是  # 也不  # 交通灯  # 就像  # 你可以  # 第一个  # 句柄  # 还会  # 自定义  # 不做 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何处理CORS跨域问题_Laravel项目CORS配置与解决方案  Windows Hello人脸识别突然无法使用  如何在香港服务器上快速搭建免备案网站?  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环  Laravel Seeder填充数据教程_Laravel模型工厂Factory使用  javascript日期怎么处理_如何格式化输出  Laravel如何处理文件下载请求?(Response示例)  如何快速搭建虚拟主机网站?新手必看指南  Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】  如何做网站制作流程,*游戏网站怎么搭建?  小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  jQuery validate插件功能与用法详解  如何快速生成ASP一键建站模板并优化安全性?  Java类加载基本过程详细介绍  英语简历制作免费网站推荐,如何将简历翻译成英文?  Android Socket接口实现即时通讯实例代码  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  智能起名网站制作软件有哪些,制作logo的软件?  如何确保西部建站助手FTP传输的安全性?  laravel怎么使用数据库工厂(Factory)生成带有关联模型的数据_laravel Factory生成关联数据方法  Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门  微信小程序 canvas开发实例及注意事项  如何在服务器上三步完成建站并提升流量?  网站制作价目表怎么做,珍爱网婚介费用多少?  Laravel如何为API编写文档_Laravel API文档生成与维护方法  Javascript中的事件循环是如何工作的_如何利用Javascript事件循环优化异步代码?  Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程  java获取注册ip实例  高防服务器租用指南:配置选择与快速部署攻略  如何续费美橙建站之星域名及服务?  手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?  javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】  如何用PHP工具快速搭建高效网站?  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程  laravel怎么配置和使用PHP-FPM来优化性能_laravel PHP-FPM配置与性能优化方法  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  进行网站优化必须要坚持的四大原则  如何在腾讯云服务器快速搭建个人网站?  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  如何确保FTP站点访问权限与数据传输安全?  javascript中对象的定义、使用以及对象和原型链操作小结  如何快速生成橙子建站落地页链接?  Laravel如何配置Horizon来管理队列?(安装和使用)  Android okhttputils现在进度显示实例代码  如何在腾讯云免费申请建站?  如何使用 Go 正则表达式精准提取括号内首个纯字母标识符(忽略数字与嵌套)