栈
1.栈内存分配从地址大的开始分配
2.栈内存中存储的数据会自动释放
3.栈的内存大小有限, 所以特别在使用递归的时候一定要注意, 栈溢出的问题
// 放在栈中的数据
int num = 666;
int value = 888;
printf("&num = %p\n", &num); // &num = 0060FEAC
printf("&value = %p\n", &value); // &value = 0060FEA8
堆
1.堆内存可以存放任意类型的数据,但需要自己申请与释放
2.堆大小,想像中的无穷大,但实际使用中,受限于实际内存的大小和内存是否连续性
- malloc
- malloc 作用:
在堆内存中申请一块我们指定长度的空间
接收一个参数: 用于指定需要申请空间的长度
会返回一个void *:
void *是万能指针, 可以转换为任意类型的指针
会将申请好的内存地址放到void *中返回给我们
- malloc 作用:
- 注意:
1.使用malloc函数必须导入一个头文件 stdlib.h
2.通过malloc申请的空间默认存储的是垃圾数据, 也就是系统不会帮我们初始化
3.所以一般情况下, malloc都会和memset函数结合起来使用
- memset作用:
专门用于初始化一块内存空间
第一个参数: 传入需要初始化内存的地址
第二个参数: 传入需要初始化的值
第三个参数: 传入需要初始化的长度
memset(void *_Dst,int _Val,size_t _Size)
- 注意点:
- 使用memset函数必须导入一个头文件 string.h
int *p = (int *)malloc(3 * sizeof(int));
// 告诉系统对那一块空间进行初始化
memset(p, 0, 3 * sizeof(int));
for(int i = 0; i < 3; i++){
printf("p[%i] = %i\n", i, p[i]);
}
int *p1 = (int *)malloc(4);
int *p2 = (int *)malloc(sizeof(int));
*p1 = 123;
*p2 = 456;
printf("*p1 = %i\n", *p1);
printf("*p2 = %i\n", *p2);
printf("p1 = %i\n", p1); // p1 = 14947416
printf("p2 = %i\n", p2); // p2 = 14947432
// 可以通过free函数来手动释放我们申请的空间
free(p1);
free(p2);
- calloc
- 就是对malloc函数的封装, 一般用于开辟数组的存储空间
- 格式:
- int *p = (int *)calloc(3, sizeof(int));//需要开辟3块存储空间, 每块存储空间占用4个字节
- realloc
- 用于给内存扩容和缩容
- 注意点:
利用malloc函数分配存储空间, 必须是连续的. 也就是在堆内存中分配内存空间必须是连续的
所以如果利用realloc函数扩容的时候, 后面没有连续的内存了, 那么系统会分配一块新的内存
int *p1 = (int *)malloc(sizeof(int));
*(p1+0) = 666;
printf("*(p1+0) = %i\n", *(p1+0));
printf("p1 = %p\n", p1);//00E325D0
// 2.给已经申请的4个字节扩容
int *p2 = (int *)realloc(p1, sizeof(int) * 2);
printf("p2 = %p\n", p2+1);//00E325D4