博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
动态分配内存
阅读量:6225 次
发布时间:2019-06-21

本文共 2000 字,大约阅读时间需要 6 分钟。

局部变量在栈中分配空间,静态变量和全局变量在静态存储区分配空间,动态内存分配,主要是在堆中分配空间。

为什么使用动态内存分配

1.声明数组时,必须在编译时知道他的长度,有些数组的长度常常在运行时才知道,若提前声明过大的数组,则会造成空间浪费。

2.当函数返回值为一个指针时,需要在函数运行时进行动态内存分配。

分配内存

其中alloca可以自动释放已分配的内存,但其他三个函数需要和free配合使用。

(1)malloc

void *malloc(size_t size);函数的参数直接是需要分配的内存字节数,返回值为一个指向被分配的内存块起始位置的指针。如果无法提供更多的内存,则返回NULL指针,因此需要对malloc返回的指针都进行检查,确保非NULL,通常和memset配合使用,一般声明如下:

1 int *pi; 2 pi = malloc(25*sizeof(int)); 3  4 cv::Point * temp = (cv::Point*)malloc(root_points.size()*sizeof(cv::Point*));////强制转换成Point* 5 if (temp == NULL){ 6     printf("Out of momery!") 7         exit(1); 8 } 9 cv::Point* ppt[1] = { temp };10 11 //cv::Point* ppt[1] = { (cv::Point*) malloc(root_points.size()) };//和4,9行的结果是一样的,malloc返回指针,因此可以直接作为指针数组的元素12 for (size_t index = 0; index < root_points.size(); index++)13     ppt[0][index] = root_points[index];14 const cv::Point* ppt_[1] = { ppt[0] };//指针数组初始化,二维数组的每一行当做其元素

 

注意:第2行中动态开辟内存时,并不是sizeof(int*),因为pi是指向整型的指针,是对pi指向的地址开辟内存,里面存储的是整型数值;如果是int **pi,则后面应该是sizeof(int*);尽量不要使用11行的方式初始化,因为无法释放,容易内存泄漏,4和9行可以对开辟的内存进行释放,更安全。

(2)calloc

void *calloc(size_t num_elements, size_t element_size);

calloc也用于分配内存,与malloc有两个区别:1.calloc分配内存后,返回指向内存的指针之前会把它初始化为0;2.函数原型不同,calloc包括所需元素数量和每个元素的字节数

(3)realloc

void *realloc(void *ptr, size_t new_size);

realloc用于修改一个原先已经分配的内存块大小,使用这个函数可以将一块内存扩大或缩小。

如果用于扩大一个内存块,该内存块原有内容保留,新增加的内存块添加到原有块的后面,若缩小一个块,该内存块的尾部部分先被拿掉,剩余内存部分原有内容保留。

若原有块无法改变大小,则realloc将分配另一块正确大小的内存,并把原有块内容复制到新块上。因此realloc后,不能使用指向旧内存的指针了,应该改用realloc返回的新指针

(4)alloca函数开辟内存后,不用free手动释放

释放内存

动态分配的内存必须整块一起释放,但realloc函数可以缩小一块动态分配的内存,释放和缩小是不一样的。

void free(void *pointer),free的参数应该是malloc、calloc、realloc返回的值,或者是NULL,如果传递是的NULL,则不会产生任何效果。

free函数的值是指针的值,而不仅仅只是申请动态内存的那个标识符,当动态开辟内存的指针复制之后,任一拷贝都可以将该内存释放,释放后,其他拷贝将不能访问该内存。

内存泄漏

内存泄漏是指内存被动态分配后,当它不再使用时未被释放,内存泄漏会增加程序的体积,有可能会导致程序或系统的崩溃,因为堆上的空间是有限的。

 结构体指针和成员指针动态分配内存

结构体中有成员为指针类型,若结构体和指针成员同时开辟了内存,在释放时,需要先释放指针成员的内存,再释放结构体指针,若反之,则无法释放指针成员的内存,从而使程序报错。另外,若只释放结构体指针,并不会释放结构体成员指针。

野指针,指向一个删除内存的但为置零的指针,或未申请内存区域的指针。

转载于:https://www.cnblogs.com/qinguoyi/p/10130440.html

你可能感兴趣的文章
SCCM 2012系列13 操作系统播发②
查看>>
Memcached 分布式缓存系统部署与调试
查看>>
开源网络备份软件bacula(功能特点与原理)
查看>>
《Essential Linux Device Drivers》第2章(下)
查看>>
Puppet扩展篇8-Puppet dashboard的部署及测试
查看>>
LCS2005客户端配置详解:LCS2005系列之二
查看>>
Linux网络管理员不得不了解的系统目录/proc/sys/net/
查看>>
如何用ACM简化你的Spring Cloud微服务环境配置管理
查看>>
自动化代码部署、代码回滚、命令执行软件之capistrano
查看>>
DHS与DSL,让生产环境更可靠
查看>>
Mac下安装Eclipse简记
查看>>
.net知识和学习方法系列(二十一)CLR-枚举
查看>>
现场感受:PowerPoint 2010轻松打造耳目一新的演讲
查看>>
挑战JavaScript正则表达式每日两题(1)
查看>>
WCF分布式开发常见错误(29):未识别的属性'targetFramework'
查看>>
Symfony2博客应用程序教程:第四部分-安全介绍
查看>>
python中if __name__ == "__main__"的解释
查看>>
《开源运营技术精髓》之负载均衡-1.2
查看>>
实践对网络安全建设思路的修正---“花瓶”模型V2.0
查看>>
如何为Linux安装Go语言
查看>>