NDK启航篇——C语言基础(内存分配)
指针、指针类型、空指针、指针运算、函数指针都介绍过了,下面来写一下内存分配
C中的内存主要分为
- 栈区(stack)
- 栈区的内存是固定的常数,如果超出了就会报
Stack OverFlow错误
,系统自动分配、释放。
- 堆区(heap)
- 堆区能够分配操作系统80%的内存,由程序员手动分配及释放。
- 全局区或静态区
- 字符常量区
- 程序代码区
这些都是我们自己做的逻辑分区,物理层面上是不存在分区的。
//栈内存 void stackFun(){ //栈内存自动释放内存 int i[1024]; } //堆内存 单位是字节1024字节为1kb,1024kb是1M void heapFun(){ //分配4M内存 malloc的返回值为void* 所以返回值类型为任意类型指针。 int *a = malloc(1024 * 1024 * sizeof(int)); //释放内存 free(a) }
内存分配分为静态内存分配
和动态内存分配
- 静态内存分配
- 编译期就确定开辟内存的大小
- 容易超出栈内存的最大值
- 容易浪费内存(为了防止内存不够而开辟更多的内存)
- 动态内存分配
- 程序运行过程中,动态开辟内存的大小,手动释放,释放后的内存可重新使用
//尖括号代表系统类库,std表示标准,io表示输入输出 引入标准输入输出 #include<stdio.h> //lib表示类库 引入标准类库 #include<stdlib.h> void main(){ //静态内存分配创建数组,数组的大小是固定的 int z = 20; int a[z]; int len; printf("请输入数组的长度:"); //创建一个数组,动态指定数组的大小(在程序运行规程中,可以随意的开辟指定大小的内存,以供使用,相当于JAVA中的集合) scanf("%d",&len); //动态开辟内存大小为len*4,p是数组的首地址也是数组的名称 int * p = malloc(len * sizeof(int)); //用刚开辟的内存区域给数组元素赋值 int i = 0; for(; i < len; i++){ p[i] = rand() % 100; printf("%d,%#xn",p[i],&p[i]); } //手动释放内存 free(p); getchar(); }
- 使用场景
(动态内存分配)
-
realloc
: realloc(原来内存的指针,内存扩大之后的总大小);
- 缩小的话会丢失缩小部分内存中的数据
- 扩大如果当前内存段后面有需要的内存空间,直接扩展这段内存空间,realloc返回原指针
- 扩大如果当前内存段后面的内存空间不够,那么使用堆中的第一个能够满足这一要求的内存块,将目前的数据复制到新的位置,并将原来的数据释放掉,返回新的内存地址
- 扩大如果申请失败,返回NULL,原来的指针仍然有效
//尖括号代表系统类库,std表示标准,io表示输入输出 引入标准输入输出 #include<stdio.h> //lib表示类库 引入标准类库 #include<stdlib.h> void main(){ //静态内存分配创建数组,数组的大小是固定的 int z = 20; int a[z]; int len; printf("请输入数组的长度:"); //创建一个数组,动态指定数组的大小(在程序运行规程中,可以随意的开辟指定大小的内存,以供使用,相当于JAVA中的集合) scanf("%d",&len); //动态开辟内存大小为len*4,p是数组的首地址也是数组的名称 int * p = malloc(len * sizeof(int)); //用刚开辟的内存区域给数组元素赋值 int i = 0; for(; i < len; i++){ p[i] = rand() % 100; printf("%d,%#xn",p[i],&p[i]); } int addLen; printf("输入数组增加的长度:"); scanf("%d",&addLen); //扩大刚分配的空间 //realloc(原来内存的指针,内存扩大之后的总大小) int * p2 = realloc(p,sizeof(int)*(len+addLen)); if(p2 == NULL){ printf("空间不足,申请失败"); } //重新赋值 i = 0; for(;i < len + addLen; i++){ p2[i] = rand() % 100; } //手动释放内存 if(p! = NULL){ free(p2); p = NULL; } if(p2 != NULL){ free(p2); p2 = NULL; } getchar(); }
内存分配注意事项
- 不能多次释放
- 释放完之后,给指针置NULL,标志释放完成
- 内存泄漏(重新赋值之前没有先释放)
好了,内存分配到这里就结束了,明天写C字符串,欲速则不达,慢慢来不着急。