指针的强大很大程度上源于它们能追踪动态分配的内存。通过指针来管理这部分内
存是很多操作的基础,包括一些用来处理复杂数据结构的操作。要完全利用这些能
力,需要理解C的动态内存管理是怎么回事。
内存的简答来说的三大操作:分配----使用----释放
内存管理指的是:分配— ----释放
我们编写的程序代码:使用
程序本质上就是处理数据,数据信息需要存放在内存里,就是用二极管表示的开断表示二进制数,进一步用二进制数表示万物:如音乐、文字、视频、图片、等等各种资源。
分配–释放:为了更好的利用和回收内存资源,最大程度的发挥计算资源,统一操作系统来调度
动态内存分配其实就要谈到自动内存分配
自动分配:内存其实就是我们在编程文件里定义的变量实际在运行时映射的内存,这部分变量的内存都是由系统自动管理(分配-释放)。函数体(或者语句块)内的变量都是分布在函数 帧 里面,运行时就会把这个帧插入到函数运行栈里,内存这些都由系统分拨,运行完就出栈,内存也被操作系统回收,一致到主函数出栈,程序退出。
动态分配:就是我们自己手动申请操作系统给我们程序分配的内存,内存区域主要在于 堆 上,这部分资源是我们手动申请和回收的。分配到的资源是我们来操作、存放数据的地方。
实际上我们定义的变量最后也会被翻译为地址,都是通过寻址来操作变量的值(可以去看看汇编语言)
#include <stdio.h> #include <stdlib.h> int main(void) { int*pi=(int*)malloc(sizeof(int)); *pi=5; printf("*pi:%d\n",*pi); free(pi); return 0; }
注意点
int *pi=(int)malloc((4));
有几个内存分配函数可以用来管理动态内存,虽然具体可用的函数取决于系统,但
大部分系统的stdlib.h头文件中都有如下函数:
函数 | 描述 |
---|---|
malloc | 从堆上分配内存 |
realloc | 在之前分配的内存块的基础上,将内存重新分配为更大或者更小的部分 |
calloc | 从堆上分配内存并清零 |
malloc函数从堆上分配一块内存,所分配的字节数由该函数唯一的参数指定,返回值是void指针,如果内存不足,就会返回NULL。此函数不会清空或者修改内存。
声明:void* malloc(size_t);
实例用法:int* pi=(int*)malloc(sizeof(int));
因为当malloc无法分配内存时会返回NULL,在使用它返回的指针之前先检查NULL是不错的做法,如下所示:
int*pi=(int*)malloc(sizeof(int)); if(pi!=NULL) { //指针没有问题 }else { //无效的指针 }
声明:void *realloc(void *ptr,size t size);
realloc函数返回指向内存块的指针。该函数接受两个参数,第一个参数是指向原内存块的指针,第二个是请求的大小。重新分配的块大小和第一个参数引用的块大小不同。返回值是指向重新分配的内存的指针。
第一个参数 | 第二个参数 | 行为 |
---|---|---|
空 | 无 | 同malloc |
非空 | 0 | 原内存块被释放 |
非空 | 比原内存块小 | 利用当前的块分配更小的块 |
非空 | 比原内存块大 | 要么在当前位置要么在其他位置分配更大的块 |
堆管理器可以重用原始的内存块,且不会修改其内容。不过程序继续使用的内存超
过了所请求的8字节。也就是说,我们没有修改字符串以便它能装进8字节的内存
块中。在本例中,我们本应该调整字符串的长度以使它能装进重新分配的8字节。
实现这一点最简单的办法是将NUL赋给地址507。实际使用的内存超出分配的内存
不是个好做法,应该避免。
calloc函数在申请内存时会清空内存【清空内存的意思是将其内容置为二进制0】
声明: void *calloc(size_t numElements,size_t elementSize);
calloc函数会根据numElements和elementSize两个参数的乘积来分配内存,并返回一个指向内存的第一个字节的指针。如果不能分配内存,则会返回NULL。此函数最初用来辅助分配数组内存。
如果numElements或elementSize为0,那么calloc可能返回空指针。如果calloc无法分配内存就会返回空指针,而且全局变量errno会设置为ENOMEM(内存不足),这是POSIX错误码,有的系统上可能没有。
有了动态内存分配,程序员可以将不再使用的内存返还给系统,这样可以释放内存
留作他用。通常用free函数实现这一点,该函数的原型如下:
声明: void free(void *ptr);
指针参数应该指向由malloc类函数分配的内存的地址,这块内存会被返还给堆。尽管指针仍然指向这块区域,但是我们应该将它看成指向垃圾数据。稍后可能重新分配这块区域,并将其装进不同的数据。
要点
如果内存已经释放,而指针还在引用原始内存,这样的指针就称为迷途指针。迷途指针没有指向有效对象,有时候也称为过早释放。
迷途指针带来的问题:
造成的原因:
//第一种情况 int*pi = (int*)malloc(sizeof(int)); printf("*pi:%d\n",*pi); free(pi); *pi = 10; //第二种情况 int*p1 = (int*)malloc(sizeof(int)); *p1 = 5; int* p2; p2 = p1; free(p1); *p2 = 10;//迷途指针 //第三种情况 /* 大部分编译器都把块语句当做一个栈帧。tmp变量分配在栈帧上,之后在块语句退出时会出栈。 pi指针现在指向一块最终可能被其他活跃记录(比如foo函数)覆盖的内存区域。 图2-13说明的就是这种情形。 */ int *pi; int tmp = 5; pi = &tmp; //这里pi变成了迷途指针 foo();