【C/C++ new/delete和malloc/free的异同及原理】

news/2024/7/7 19:48:49 标签: c语言, c++, java

new/delete和malloc/free都是用于在C++(以及C语言在malloc/free的情况下)中动态申请和释放内存的机制,但它们之间存在一些显著的异同点。以下是对这两组函数/运算符的异同点的详细分析:

相同点

  1. 目的相同:两者都用于在堆(heap)上动态地分配和释放内存。
  2. 手动管理:无论是new/delete还是malloc/free,都需要程序员手动进行内存的分配和释放,以避免内存泄漏。

不同点

  1. 类型与来源
    new/delete:是C++的运算符,支持重载,可以与C++的类特性(如构造函数、析构函数)紧密集成。
    malloc/free:是C语言(及C++兼容)的标准库函数,分别定义在<stdlib.h>(C语言)和(C++)头文件中。
  2. 内存分配与初始化
    new:能够自动根据对象的类型计算所需内存大小,并在分配内存后自动调用对象的构造函数进行初始化(如果是类类型的对象)。
    malloc:需要程序员手动计算所需内存大小,并且分配的内存不会自动初始化,通常包含未定义的值。
  3. 返回值
    new:返回指向分配的内存的指针,该指针的类型是所分配对象的类型。
    malloc:返回void*类型的指针,指向分配的内存,需要程序员根据需要进行类型转换。
  4. 错误处理
    new:如果内存分配失败,会抛出std::bad_alloc异常。
    malloc:如果内存分配失败,会返回NULL(在C++11及以后,推荐使用nullptr)。
  5. 内存释放
    delete:在释放内存时,会调用对象的析构函数(如果是类类型的对象),然后释放内存。
    free:仅释放之前通过malloc分配的内存,不会调用任何析构函数。
  6. 数组支持
    new/delete:对于数组,有专门的语法new 类型[大小]和delete[] 指针来分配和释放内存。
    malloc/free:对于数组,只需在malloc中指定数组总大小(每个元素的类型大小乘以元素数量),并在free中传递指向数组首元素的指针。但注意,malloc不会自动处理数组元素的构造或析构。
  7. 类型安全
    new:提供了更好的类型安全,因为返回的是具体类型的指针,编译器可以进行类型检查。
    malloc:由于返回void*类型,需要程序员进行类型转换,这可能引入类型错误。
  8. 性能
    在底层实现上,new/delete通常是对malloc/free的封装,因此它们的性能差异主要取决于封装层的开销。在大多数情况下,这种差异可以忽略不计,但在性能敏感的应用中可能需要考虑。
现在开始添加阿秀总结部分(很清晰、易懂,非常建议学习
  1. new是类型安全的,malloc不是。例如:

    int*p=newfloat[2];//编译错误
    int *p = (int*)malloc(2*sizeof(double));//编译无错误
    
  2. new调用名为operator new的标准库函数分配足够空间并调用相关对象的构构造函数,delete对指针所指对象运行适当的析构函数,然后通过调用名为perator delete的标准库函数释放该对象所用内存。后者均没有相关调用。

  3. new是封装了malloc,直接free不会报错,但是这只是释放内存,而不会析构对象

  4. new和delete是如何实现的?
    new的实现过程是:首先调用名为operator new的标准库函数,分配足够大的原始为类型化的内存,以保存指定类型的一个对象;接下来运行该类型的一个构造函数,用指定初始化构造对象;最后返回指向新分配并构造后的的对象的指针;
    delete的实现过程:对指针指向的对象运行适当的析构函数;然后通过调用名为operator delete的标准库函数释放该对象所用内存;

  5. malloc和new的区别?
    malloc和free是标准库函数,支持覆盖;new和delete是运算符,支持重载。
    malloc仅仅分配内存空间,free仅仅回收空间,不具备调用构造函数和析构函数功能,用malloc分配空间存储类的对象存在风险;new和delete除了分配回收功能外,还会调用构造函数和析构函数。
    malloc和free返回的是void类型指针(必须进行类型转换),new和delete返回的是具体类型指针。

  6. 既然有了malloc/free,C++中为什么还需要new/delete呢?直接用malloc/free不好吗?
    malloc/free和new/delete都是用来申请内存和回收内存的。
    在对非基本数据类型的对象使用的时候,对象创建的时候还需要执行构造函数,销毁的时候要执行析构函数。而malloc/free是库函数,是已经编译的代码,所以不能把构造函数和析构函数的功能强加给malloc/free,所以new/delete是必不不可少的。

  7. 被free回收的内存是立即返还给操作系统吗?
    不是的,被free回收的内存会首先被ptmalloc使用双链表保存起来,当用户下一次申请内存的时候,会尝试从这些内存中寻找合适的返回。这样就避免了频繁的系统调用,占用过多的系统资源。同时ptmalloc也会尝试对小块内存进行合并,避免过多的内存碎片。

总结

new/delete和malloc/free各有其适用场景。在C++程序中,由于new/delete能够与C++的类特性更好地集成,因此通常是首选的内存分配方式。然而,在处理与C语言的接口或需要更低级内存操作的情况下,malloc/free仍然有其不可替代的作用。重要的是,无论使用哪种方式,都应确保及时释放分配的内存,以避免内存泄漏。


http://www.niftyadmin.cn/n/5535021.html

相关文章

quill编辑器使用总结

一、vue-quill-editor 与 quill 若使用版本1.0&#xff0c;这两个组件使用哪个都是一样的&#xff0c;无非代码有点偏差&#xff1b;若需要使用表格功能&#xff0c;必须使用 quill2.0 版本&#xff0c;因为 vue-quill-editor 不支持table功能。 二、webpack版本问题 在使用 q…

xinput1_4.dll丢失怎么办?如何来解决xinput1_4.dll丢失问题

在电脑启动游戏时候经常会出现一些问题导致游戏无法打开运行&#xff0c;其中找不到xinput1_4.dll文件丢失就是常见问题之一&#xff0c;那么当遇到xinput1_4.dll丢失怎么办呢&#xff1f;今天就教大家如何来解决xinput1_4.dll丢失问题。 一、xinput1_4.dll文件详解 XINPUT1_4…

spdlog一个非常好用的C++日志库(四): 源码分析之logger类

目录 1.简介 2.类图关系 3.logger数据成员 4.logger函数成员 4.1.构造与析构 4.1.1.构造函数 4.1.2.拷贝构造、移动构造 4.2.交换操作 4.3.log()记录日志消息 4.3.1.格式串 4.3.2.普通字符串 4.3.3.日志级别 4.3.4.宽字符支持 4.4.sink_it_&#xff1a;将log消息…

LabVIEW新能源汽车电池性能测试系统

新能源汽车的核心部件之一是电池&#xff0c;其性能直接关系到整车的续航里程、安全性和寿命。为了确保电池的性能和可靠性&#xff0c;测试是必不可少的环节。本文介绍了一种基于LabVIEW的新能源汽车电池性能测试系统&#xff0c;通过LabVIEW与数据采集设备的无缝集成&#xf…

FreeRTOS学习 -- 软件定时器

一、软件定时器简介 1、软件定时器概述 软件定时器允许设置一段时间&#xff0c;当设置的时间到达之后就执行指定的功能函数&#xff0c;被定时器调用的这个功能函数叫做定时器的回调函数。回调函数的两次执行间隔叫做定时器的定时周期。 简而言之&#xff0c;当定时器的定时…

什么是嵌入式,单片机又是什么,两者有什么关联又有什么区别?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「嵌入式的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;从科普的角度&#xff0c;…

PostgreSQL LIMIT 子句

PostgreSQL LIMIT 子句 PostgreSQL 是一种功能强大的开源对象关系数据库管理系统&#xff0c;广泛用于各种应用中。在处理大量数据时&#xff0c;我们通常只需要检索部分记录&#xff0c;而不是整个数据集。这时&#xff0c;LIMIT 子句就变得非常有用。本文将详细介绍 Postgre…

字典树模板+位运算

P3879 [TJOI2010] 阅读理解 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) trie树板子题&#xff0c;稍微有一丢丢不一样&#xff0c;套用字典树模板稍加修改就能过 手搓字典树代码&#xff1a; char ch[1010][26], cnt[1010], idx; void insert(string s)//插入 {int p 0…