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

C语言指针之二malloc的用法及详解

一、为什么c语言中要有malloc

malloc就是memory allocate动态分配内存,malloc的出现时为了弥补静态内存分配的缺点,静态分配内存有如下缺点:

1、比如说,传统的一维数组,如int a,使用传统的一维数组需要事先指定数组的长度,而且数组的长度必须是一个常量(宏定义的 常量)

2、传统数组(静态分配),不能手动释放,只能等待系统释放,静态分配的变量在该函数内运行的时候有效,当静态分配的变量所在函数运行完之后,该内存会自动释放。静态分配的内存,是在栈中分配的,其实在C语言中的函数调用也是通过栈来实现的,栈这种数据结构的一个特点就是(先进后出),所以,在调用函数的时候,都是先压入栈中,然后,再从最上面的函数开始执行,最后,执行到main函数结束。动态分配通过malloc分配,是在堆中分配的,堆不是一种数据结构,它是一种排序方式,堆排序。

3、传统数组的长度一旦定义之后,就不能更改,比如说,如果我有一个业务在这之前给分配的大小为100,但是,我现在由于业务数量的增长,原来的大小就无法满足。

4、静态分配不能跨函数调用,就是无法在另一个函数中,来管理一个函数中的内存。静态分配,只在当前函数有效,当,静态分配所在的函数运行完之后,该变量就不能被其他的函数所调用。

二、malloc是什么

malloc其实就是一个可以动态分配内存的函数,从而可以很好的弥补上面静态分配的缺点。

三、malloc怎么使用

1、使用malloc函数的时候,需要包含一个头文件#include

2、malloc函数只接受一个形参如,int *p = (int *)malloc(sizeof(int)).先来解释下这句话的含义,int* p代表一个以int类型地址为内容的指针变量,p这个变量占4个字节(某些计算机),这个p变量是静态分配的一个变量。在某些计算机的前提下,指针变量所占的大小都是一样的,无论是char* 还是long *,因为,这些指针变量里面存放的是一个8位16进制的地址,所以占四个字节,当然这些都是在某些计算机的前提下,并不是所有的都是这样的。说道地址的话,就和计算机的地址总线有关,如果计算机的地址总线是32根,每根地址总线只有两种状态(1或0),32根地址线的话,如果全为1的话,刚好就是一个8位十六进制,一位十六进制等于四个二进制(2^4=16)。32根地址总线可以 表示2^10*2^10*2^10*2^2种状态,可以表示的最大内存为4G,也就是说32根地址总线(也就是四个字节 的指针变量)最大可以表示4G内存。malloc函数会返回开辟空间的首地址,加(int *)的目的是让计算机知道,如何去划分这个开辟的空间,因为char、int 、long这些类型的字节大小是不一样的,我们知道了首地址,还要知道是以几个字节为单元。所以,这句话一共开辟了8个字节(某些计算机上),这也是为什么我写sizeof(int),而不是直接写4的原因。

3、malloc开辟空间所返回的首地址是动态分配的。

四、malloc函数的使用例子

# include # include int main(void) { int i = 5; //分配了4个字节 静态分配 int * p = (int *)malloc(sizeof(4));//指针变量p为静态分配,malloc开辟的空间为动态分配 *p = 5; //*p 代表的就是一个int变量,指针变量p表示是一个以int类型的地址为内容的变量 free(p); //freep(p)表示把p所指向的内存给释放掉,p本身的内存是静态的,不能由程序员手动释放 //p本身的内存只能在p变量所在的函数运行终止时由系统自动释放 return 0; }
# include # include void f(int * q) { *q = 200; //free(q); //把q所指向的内存释放掉,不然后面在使用*p的时候会报错,因为p所指向的内容已经被释放了 } int main(void) { int * p = (int *)malloc(sizeof(int)); //sizeof(int)返回值是int所占的字节数 *p = 10; f(p); //p是int *类型 printf("%d\n", *p); return 0; }
http://www.cnnetsun.cn/news/2815657.html

相关文章:

  • 单人创业,靠 StarLny 搭建数字团队
  • 避坑指南:ABAP里同时调用WS_REVERSE_GOODS_ISSUE和BAPI_OUTB_DELIVERY_CHANGE报VL216错误的深层原因与替代方案
  • Infra CONVERT 德国标准下的图纸自动化识别与检验计划生成指南
  • 完全免费的Android开源相机神器:OpenCamera专业摄影指南
  • 【stack、queue、deque、priority_queue】C++ 栈 / 队列 / 优先级队列全解析!手撕实现 + 二叉树层序遍历(附源码)
  • KMS_VL_ALL_AIO:Windows与Office批量激活的终极技术方案
  • 保姆级教程:用FNL数据从零搭建WRF环境并成功运行第一个案例(避坑指南)
  • 告别phpMyAdmin!一个Docker容器搞定MySQL、PostgreSQL、MongoDB,Adminer保姆级安装与多数据库连接实战
  • Windows 10/11 下用 Visual Studio 2019 编译 ZLMediaKit 流媒体服务,保姆级避坑指南
  • 信号处理实战:用db4小波分析你的传感器数据(MATLAB验证+C语言移植指南)
  • AI人脸识别考勤签到系统
  • 别再手动整理BOM了!用Excel自定义Altium Designer料单模板,效率翻倍(附模板文件)
  • 【闲聊】孩子越长大为什么越不愿意和父母讲心里话(亿点不一样)
  • 第【7】期--自由空间光通信(FSO)在Gamma-Gamma湍流信道下的BER性能仿真-maltab完整代码+报告
  • 零基础落地!三个精益实操技巧,激活员工主动改善意识
  • 别再死记硬背了!一张图+Python脚本帮你彻底搞懂ISO15765-2网络层多帧传输与流控
  • STM32H743ZI驱动DP83848实现网线热插拔:从硬件中断到lwip 2.1.3链路状态管理的完整流程
  • 用CODESYS仿真一个真实的冰箱:从ST代码反推PLC控制逻辑设计
  • STM32H743ZI驱动DP83848,从硬件连线到lwip2.1.3协议栈移植的保姆级避坑指南
  • Cursor 高级指南(二):Agent、Plan、Ask、Debug 与 Tab、内联编辑
  • 10|Netty native epoll 与零拷贝:从 Java NIO 再往下看一层![
  • Cherry Studio缺失instructions导致OpenAI-Response API访问失败
  • 大千万级文档 RAG,这 11 个步骤把幻觉压到极低
  • 分布式存储架构设计与一致性算法实践
  • Qt 入门 09|Qt 常用容器:QString/QByteArray/QList/QVector 字符串与容器使用大全
  • 终极JSXBIN解码器指南:快速解密Adobe ExtendScript二进制文件
  • Spring AI 从入门到精通-ChatClient你与 AI 对话的终极武器
  • 神经渲染:重塑室内设计的“造梦引擎”——从原理到落地全解析
  • 深度解析Jsxer:JSXBIN二进制反编译引擎的架构设计与实现原理
  • 终极macOS清理指南:使用Pearcleaner彻底告别应用残留文件