热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C++中的malloc函数详解

malloc是C语言中的一个标准库函数,全称为memoryallocation,即动态内存分配。它用于在程序运行时申请一块指定大小的连续内存区域,并返回该区域的起始地址。当无法预先确定内存的具体位置时,可以通过malloc动态分配内存。

malloc 的全称是 memory allocation,中文译为动态内存分配。它用于在程序运行时申请一块指定大小的连续内存区域,并返回该区域的起始地址。当无法预先确定内存的具体位置时,可以通过 malloc 动态分配内存,且分配的大小就是程序要求的大小。

函数定义

malloc 的函数原型为:void *malloc(size_t size); 其作用是在内存的动态存储区中分配一个长度为 size 的连续空间。此函数的返回值是分配区域的起始地址,即一个指针,指向该分配域的开头位置。

如果分配成功,则返回指向被分配内存的指针(该存储区中的初始值不确定),否则返回空指针 NULL。当内存不再使用时,应使用 free() 函数将内存块释放。函数返回的指针需要适当对齐,以便可以用于任何数据对象。

早期版本的 malloc 返回 char 类型的指针,但新的 ANSI C 标准规定,该函数返回 void 类型的指针。因此,在使用时可能需要进行类型转换。malloc 能向系统申请分配一个长度为 num_bytes(或 size)个字节的内存块。通常,malloc 需要与 free 函数配对使用,以确保动态分配的内存能够正确释放。

工作机制

malloc 函数的核心机制在于维护一个空闲链表,该链表将所有可用的内存块连接起来。当调用 malloc 函数时,它会沿着链表寻找一个足够大的内存块来满足用户的请求。找到合适的内存块后,将其一分为二:一块与用户请求的大小相等,另一块则是剩余的部分。然后,将分配给用户的内存块返回给用户,并将剩余的内存块(如果有的话)重新放回链表中。调用 free 函数时,它会将用户释放的内存块重新连接到空闲链表上。

随着时间的推移,空闲链表可能会被分割成许多小的内存片段。如果用户请求一个较大的内存片段,而空闲链表中没有足够的连续内存块,malloc 会尝试将相邻的小内存片段合并成较大的内存块。如果仍然无法找到合适的内存块,malloc 将返回 NULL 指针。因此,在调用 malloc 动态申请内存时,必须检查其返回值是否为 NULL。

在 Linux 系统中,glibc 库通过 brk() 和 mmap() 系统调用来实现 malloc。当释放内存时,glibc 会尝试整合相邻的内存碎片,以提高内存利用率。

与 new 的区别

从本质上讲,malloc 是 C 语言标准库中的一个函数,而 new 是 C++ 语言中的关键字。malloc 需要包含头文件 stdlib.h,而 new 不需要包含任何头文件。C++ 编译器可以直接将 new 编译为目标代码,并插入相应的构造函数。

在使用上,malloc 和 new 有以下主要区别:

  • new 返回指定类型的指针,并且可以自动计算所需内存的大小。而 malloc 需要手动计算字节数,并在返回后强制转换为实际类型的指针。
  • malloc 只负责分配内存,不会对内存进行初始化,因此新分配的内存中的值是随机的。而 new 除了分配内存外,还会调用对象的构造函数,对内存进行初始化。
  • 通过 malloc 或 new 分配的内存,在其他操作上保持一致,但在释放内存时,malloc 使用 free 函数,而 new 使用 delete 运算符。

示例代码

以下是一个简单的示例,演示如何使用 malloc 函数分配和释放内存:

#include 
#include 

int main() {
    char *p;
    
    p = (char *)malloc(100);
    if (p) {
        printf("Memory Allocated at: %p\n", p);
    } else {
        printf("Not Enough Memory!\n");
    }
    free(p);
    return 0;
}

推荐阅读
  • 在1995年,Simon Plouffe 发现了一种特殊的求和方法来表示某些常数。两年后,Bailey 和 Borwein 在他们的论文中发表了这一发现,这种方法被命名为 Bailey-Borwein-Plouffe (BBP) 公式。该问题要求计算圆周率 π 的第 n 个十六进制数字。 ... [详细]
  • 二维码的实现与应用
    本文介绍了二维码的基本概念、分类及其优缺点,并详细描述了如何使用Java编程语言结合第三方库(如ZXing和qrcode.jar)来实现二维码的生成与解析。 ... [详细]
  • 本文介绍了如何通过C#语言调用动态链接库(DLL)中的函数来实现IC卡的基本操作,包括初始化设备、设置密码模式、获取设备状态等,并详细展示了将TextBox中的数据写入IC卡的具体实现方法。 ... [详细]
  • 数据类型--char一、char1.1char占用2个字节char取值范围:【0~65535】char采用unicode编码方式char类型的字面量用单引号括起来char可以存储一 ... [详细]
  • 本文将从基础概念入手,详细探讨SpringMVC框架中DispatcherServlet如何通过HandlerMapping进行请求分发,以及其背后的源码实现细节。 ... [详细]
  • 如何在PHP中安装Xdebug扩展
    本文介绍了如何从PECL下载并编译安装Xdebug扩展,以及如何配置PHP和PHPStorm以启用调试功能。 ... [详细]
  • importjava.io.*;importjava.util.*;publicclass五子棋游戏{staticintm1;staticintn1;staticfinalintS ... [详细]
  • 解决Visual Studio Code中PHP Intelephense误报问题
    PHP作为一种高度灵活的编程语言,其代码结构可能导致Intelephense插件在某些情况下报告不必要的错误或警告。自1.3.3版本起,Intelephense引入了多个配置选项,允许用户根据具体的工作环境和编程风格调整这些诊断信息的显示。 ... [详细]
  • 嵌套列表的扁平化处理
    本文介绍了一种方法,用于遍历嵌套列表中的每个元素。如果元素是整数,则将其添加到结果数组中;如果元素是一个列表,则递归地遍历这个列表。此方法特别适用于处理复杂数据结构中的嵌套列表。 ... [详细]
  • 本文介绍了SIP(Session Initiation Protocol,会话发起协议)的基本概念、功能、消息格式及其实现机制。SIP是一种在IP网络上用于建立、管理和终止多媒体通信会话的应用层协议。 ... [详细]
  • 本文详细介绍了C++中的构造函数,包括其定义、特点以及如何通过构造函数进行对象的初始化。此外,还探讨了转换构造函数的概念及其在不同情境下的应用,以及如何避免不必要的隐式类型转换。 ... [详细]
  • 本文详细介绍了iOS应用的生命周期,包括各个状态及其转换过程中的关键方法调用。 ... [详细]
  • Windows操作系统提供了Encrypting File System (EFS)作为内置的数据加密工具,特别适用于对NTFS分区上的文件和文件夹进行加密处理。本文将详细介绍如何使用EFS加密文件夹,以及加密过程中的注意事项。 ... [详细]
  • 在处理大数据量的SQL分页查询时,通常需要执行两次查询来分别获取数据和总记录数。本文介绍了一种优化方法,通过单次查询同时返回分页数据和总记录数,从而提高查询效率。 ... [详细]
  • 本文通过一个具体的实例,介绍如何利用TensorFlow框架来计算神经网络模型在多分类任务中的Top-K准确率。代码中包含了随机种子设置、模拟预测结果生成、真实标签生成以及准确率计算等步骤。 ... [详细]
author-avatar
起薪d这帖
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有