既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
👋静态内存开辟₀
再说动态内存开辟的时候,我们先来介绍下我们已经学会的开辟内存空间的方法吧。
char arr[10] = {0}; //在所处的栈上连续开辟10个字节的内存空间
int a = 1; //在栈上开辟4个字节空间
这些就是我们前面所学的知识点常用开辟内存空间的办法↓
- 我们所开辟的内存空间大小是固定的。
- 我们在申明数组的时候,必须要指定数组当中的长度,这样它所需要的内存空间才能被编译系统所知道。从而得到内存上的一个分配。
但是这样就会导致一个问题,当我们有的时候需要的空间大小必须是要在程序运行之后才能够被知道,那用数组的编译时候开辟空间的方式就不能够被满足。
这个就好比就是我想要在一个程序运行当中当中输入两百个同学的信息,此时我用数组分配了1000个同学的信息。此时我就会浪费很多的内存空间。当我想输入2000个同学信息的时候,此时数组分配的1000个同学信息存放不下去。
那么这个时候我们就可以使用动态内存开辟就可以完美的解决这个问题了。
👋动态内存开辟¹
什么是动态内存开辟?
所谓动态内存分配(Dynamic Memory Allocation)就是指在程序执行的过程中动态地分配或者回收存储空间的分配内存的方法。动态内存分配不象数组等静态内存分配方法那样需要预先分配存储空间,而是由系统根据程序的需要即时分配,且分配的大小就是程序要求的大小。
那么既然说到这个再来说说栈和堆的概念n:↓这样更加方便且容易理解。
👌栈
在执行函数的时候,函数内部局部变量的存储单元都是可以在栈上进行创建的,函数执行结束的时候这些存储单元会被自动的进行释放。栈区主要存放运行函数所分配的局部变量,函数的参数,返回数据,返回地址等。
👌堆
通常定义变量(或对象),编译器在编译时都可以根据该变量(或对象)的类型知道所需内存空间的大小,从而系统在适当的时候为他们分配确定的存储空间。这种内存分配称为静态存储分配;有些操作对象只在程序运行时才能确定,这样编译时就无法为他们预定存储空间,只能在程序运行时,系统根据运行时的要求进行内存分配,这种方法称为动态存储分配。所有动态存储分配都在堆区中进行。
当程序运行到需要一个动态分配的变量或对象时,必须向系统申请取得堆中的一块所需大小的存储空间,用于存贮该变量或对象。当不再使用该变量或对象时,也就是它的生命结束时,要显式释放它所占用的存贮空间,这样系统就能对该堆空间进行再次分配,做到重复使用有限的资源。
接下来介绍下动态内存开辟的函数的使用,知道如何用了,那么就相当于学会了这个动态内存开辟。
👏malloc()函数
malloc()函数的声明,如下所示↓
void* malloc (size_t size);
分配一个大小为字节的内存块,返回一个指向该块开头的指针。
新分配的内存块的内容没有初始化,剩下不确定的值。
如果size为0,则返回值取决于特定的库实现(它可能是一个空指针,也可能不是),但返回的指针不应被解引用。
size→内存块的大小,以字节为单位。
- Size_t是一个无符号整型类型。
成功时,指向函数分配的内存块的指针。
该指针的类型始终是void*,可以将其转换为所需类型的数据指针,以便可解引用。
如果函数分配请求的内存块失败,则返回一个空指针NULL。
使用 malloc() 函数需要引头文件 #include<stdlib.h>
👏free()函数
free()函数的声明,如下所示↓
void free (void* ptr);
ptr → 指针指向一个要释放内存的内存块,该内存块之前是通过调用 malloc、calloc 或 realloc 进行分配内存的。如果传递的参数是一个空指针,则不会执行任何动作。
该函数不返回任何值。
实际上 free() 函数是专门用来作动态内存的释放或者回收的。注意:free()当中的 ptr 是指向栈区的空间的,如果你存放堆区空间的话是属于未定义的。
当我们回收完成之后,最好把 ptr 当中的参数置成为 NULL,这样它就找不到那块原来的空间了。
使用 free() 函数需要引头文件 #include<stdlib.h>
👏malloc()动态内存开辟使用
示例代码如下所示↓
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int* p = (int*)malloc(10 * sizeof(int));
if (p == NULL)
{
perror("main:");//perror() - 打印错误信息
}
int i = 0;
for (i = 0; i < 10; i++)
{
*(p + i) = i;//初始化赋值
printf("%d\n", p[i]);//相当于数组打印元素
}
//回收内存空间
free(p);
p = NULL;
return 0;
}
运行结果如下↓
0 1 2 3 4 5 6 7 8 9
有没有发现其实和数组的访问是一模一样的,只不过一个是在栈区上,这个是在堆区上。
那如果是 malloc() 函数返回的是空指针的情况n,我们只需要把malloc()内存块空间改变下即可。如下↓
int* p = (int*)malloc(1000000000000000 * sizeof(int));
那么此时的运行结果就是→Not enough space(没有足够的空位)是实际上也就是内存空间。
👏calloc()
calloc()函数的声明,如下所示↓
void* calloc (size_t num, size_t size);
为num元素数组分配一块内存,每个元素大小为字节长,并将其所有位初始化为零。有效的结果是分配一个(num*size)字节的零初始化内存块。
如果size为0,则返回值取决于特定的库实现(它可能是一个空指针,也可能不是),但返回的指针不应被解引用。
- 参数
num → 要分配的元素数量。
size → 每个元素的大小。
size_t→ 是无符号整型。
- 返回值
成功时,指向函数分配的内存块的指针。
该指针的类型始终是void*,可以将其转换为所需类型的数据指针,以便可解引用。
如果函数分配请求的内存块失败,则返回一个空指针。
calloc()函数和malloc()函数最大的区别无非就是:calloc()比malloc()就是参数多了一个,以及calloc()的函数会把每个字节进行初始化为全0,而malloc()不会对每个字节的内容进行初始化。calloc()会在返回地址之前把申请的空间的每个字节初始化全0的。
👏calloc()动态内存开辟使用
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
化为全0,而malloc()不会对每个字节的内容进行初始化。calloc()会在返回地址之前把申请的空间的每个字节初始化全0的。
👏calloc()动态内存开辟使用
[外链图片转存中…(img-61AvBCuA-1715850895122)]
[外链图片转存中…(img-qiMuqwVM-1715850895122)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新