NDK启航篇——C语言基础(内存分配)

ndk

NDK启航篇——C语言基础
NDK启航篇——C语言基础(指针)
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,%#x\n",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,%#x\n",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字符串,欲速则不达,慢慢来不着急。

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 首先我们来科普一下: 什么是堆?说到堆,又忍不住说到了栈!什么是 栈?1、什么是堆:堆是大家共有的空间,分全局堆和...
    Xiho丶阅读 5,011评论 0 1
  • 很多新学C语言的童鞋在用到动态内存分配的时候,对选择哪种分配函数及其有何区别搞不清楚,那么下文就认真的讲讲它们的种...
    Leon_Geo阅读 4,693评论 0 2
  • Android NDK开发之旅 目录 C 内存管理函数 C 语言为内存的分配和管理提供了几个函数。这些函数可以在 ...
    香沙小熊阅读 5,826评论 1 0
  • 逻辑上的分区 栈区 堆区 静态区 常量区 代码区 代码区,常量区,静态区,堆区,栈区这个排列顺序按照地址由小到大排...
    MathCat阅读 3,777评论 1 0
  • 首先,我们应该知道。所有的程序都必须留出足够的内存空间来存储所使用的数据,所以我们常常会预先给程序开辟好内存空间,...
    發財大力阅读 1,684评论 0 1

友情链接更多精彩内容