首页 > 系统相关 >零基础非科班也能掌握的C语言知识19 动态内存管理

零基础非科班也能掌握的C语言知识19 动态内存管理

时间:2024-06-09 17:59:46浏览次数:28  
标签:malloc 19 free C语言 int 内存 动态内存 开辟

动态内存管理

1.为什么要有动态内存分配

我们目前掌握的内存开辟方式仅仅是在栈上开辟一个大小固定好的空间。但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间⼤⼩在程序运⾏的时候才能知道,那在栈上开辟空间的⽅式就不能满⾜了。(尽管我们会通过变长数组的方式开辟栈空间,但是变长数组的大小在确定后就不能改变了),我们需要让程序员⾃⼰可以申请和释放空间,因此C语言引入了动态内存开辟

2.malloc和free

2.1 malloc

C语⾔提供了⼀个动态内存开辟的函数:malloc
在这里插入图片描述

在这里插入图片描述
注意malloc的参数是size_t的类型,开辟的空间是以字节为单位的。

2.2 free

C语⾔提供了另外⼀个函数free,专⻔是⽤来做动态内存的释放和回收的,函数原型如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
注意free释放了动态开辟的内存的之后,应及时将ptr置为空指针否则的话ptr将变成野指针。往往free和置为空指针是同时出现的

3.calloc和realloc

3.1 calloc

在这里插入图片描述

在这里插入图片描述
例子:
在这里插入图片描述

3.2realloc

在这里插入图片描述

在这里插入图片描述
但是动态开辟内存是可能失败的也及时说,情况二下我们开辟动态内存失败会造成原数据丢失,因此我们应该选择代码2。
在这里插入图片描述
当然在没有开辟过动态内存时我们也可以用realloc来开辟

#include<stdlib.h>
int main()
{
realloc(NULL,10);
return 0;
}

4.常见的动态内存的错误

4.1对NULL指针的解引用操作

因此开辟内存空间(可能开辟失败)后一定要判断是否返回的是一个空指针

4.2对动态开辟空间的越界访问

#include<stdlib.h>
void test()
{
	int i = 0;
	int* p = (int*)malloc(10 * sizeof(int));
	if (NULL == p)
	{
		exit(EXIT_FAILURE);
	}
	for (i = 0; i <= 10; i++)
	{
		*(p + i) = i;//当i是10的时候越界访问
	}
	free(p);
}
int main()
{
	test();
}

在这里插入图片描述

4.3对非动态内存开辟的空间free

#include<stdlib.h>
void test()
{
	int a = 10;
	int* p = &a;
	free(p);//ok?
 }
int main()
{
	test();
}

在这里插入图片描述

4.4使用free释放⼀块动态开辟内存的⼀部分

#include<stdlib.h>
void test()
 {
 int *p = (int *)malloc(100);
 p++;
 free(p);//p不再指向动态内存的起始位置
 }

4.5对同⼀块动态内存多次释放

#include<stdlib.h>
void test()
 {
 int *p = (int *)malloc(100);
 free(p);
 free(p);//重复释放
 }

4.6动态开辟内存忘记释放(内存泄漏)

#include<stdlib.h>
void test()
 {
 int *p = (int *)malloc(100);
 if(NULL != p)
 {
 *p = 20;
 }
 }
int main()
 {
 test();
 while(1);
 }

感觉遇到这个问题是最棘手的,因为一开始程序是正常运行的。然后内存泄漏堆区占满了,程序直接崩溃了。关键这个问题还不好察觉,因此动态内存开辟一定要free并且置零

5.柔性数组

也许你从来没有听说过柔性数组(flexible array)这个概念,但是它确实是存在的。
C99 中,结构中的最后⼀个元素允许是未知⼤⼩的数组,这就叫做『柔性数组』成员。
在这里插入图片描述

5.1柔性数组的特点

在这里插入图片描述

5.2柔性数组的使用

#include<stdlib.h>
typedef struct st_type
{
	int i;
	int a[0];//柔性数组成员
}type_a;
int main()
{
	int i = 0;
	type_a* p = (type_a*)malloc(sizeof(type_a) + 100 * sizeof(int));
	//业务处理
	p->i = 100;
	for (i = 0; i < 100; i++)
	{
		p->a[i] = i;
	}
	free(p);
	return 0;
}

6.总结C/C++中程序内存区域划分

在这里插入图片描述

标签:malloc,19,free,C语言,int,内存,动态内存,开辟
From: https://blog.csdn.net/Yusei_0523/article/details/139534202

相关文章

  • 实验6 C语言结构体、枚举应用编程
    #defineN3//运行程序输入测试时,可以把这个数组改小一些输入测试#include<stdlib.h>typedefstructstudent{intid;//学号charname[20];//姓名charsubject[20];//考试科目doubleperf;//平时成绩......
  • 数据结构严蔚敏版精简版-线性表以及c语言代码实现
    线性表、栈、队列、串和数组都属于线性结构。线性结构的基本特点是除第一个元素无直接前驱,最后一个元素无直接后继之外,其他每个数据元素都有一个前驱和后继。1 线性表的定义和特点如此类由n(n大于等于0)个数据特性相同的元素构成的有限序列称为线性表。线性表中元素的个数n定......
  • [题解]P1967 [NOIP2013 提高组] 货车运输
    P1967[NOIP2013提高组]货车运输题意简述给定一个\(N\)个节点,\(M\)条边的无向图,其中每条边有一个边权。接下来给定\(q\)次询问。每次询问给出\(x,y\),请计算\(x\)到\(y\)路径上最小边权的最大值是多少。解题思路我们对于每个连通块跑一遍最大生成树。这样整张图就成了一片森......
  • C语言数据类型和变量
    1.数据类型介绍c语言提供了各种数据类型来描述生活中的各种数据。下面介绍一下c语言内置数据类型:1.1字符型char//charactersignedchar//有符号的unsignedchar//⽆符号的1.2整型//短整型shortintsignedshortintunsignedshortint//整型intsign......
  • 二维数组2(第2~5题是Goc语言,其余是C++语言)
    第1题    对角线    时限:1s空间:256m输入整数N,输出相应方阵。输入格式一个整数N。(0<n<10)输出格式一个方阵,每个数字的场宽为3。输入/输出例子1输入:5输出: 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 ......
  • C语言王国——数据的内存管理
    目录一、引言二、整形在内存中的存储2.1进制之间的转换2.1.1整形的二进制2.1.2 十进制和二进制2.1.3十进制和八进制的转换2.1.4 十六进制和十进制的转换2.2原码,反码,和补码三、大、小端字节序3.1大小端的定义3.2为什么会有大小端之分3.3代码区分四、浮......
  • (C语言)常见字符函数和字符串函数(详解)
    我们都知道,在C语言里有string.h这个头文件,但是C语言里没有string这个类型。字符串通常放在常量字符串中或者字符数组中,字符串常量适用于那些对她不做修改的字符串函数。string.h这个头文件里声明的函数原型也全是针对char数组的种种操作。直到C++中才出现了string这个类这......
  • 实验6 C语言结构体、枚举应用编程
    //task4.c#include<stdio.h>#include<stdlib.h>#include<string.h>#defineN10typedefstruct{charisbn[20];//isbn号charname[80];//书名charauthor[80];//作者doublesales_price;//售价int......
  • 实验6 C语言结构体、枚举应用编程
    task.1#defineN3//运行程序输入测试时,可以把这个数组改小一些输入测试#include<stdlib.h>typedefstructstudent{intid;//学号charname[20];//姓名charsubject[20];//考试科目doubleperf;//......
  • 表达式求值的相关语法知识(C语言)
    目录整型提升整型提升的意义整型提升规则整型提升实例算术转换赋值转换操作符的属性C语言的语法并不能保证表达式的执行路径唯一!!!问题表达式整型提升        C的整型算术运算总是至少以缺省整型类型的精度来进行的。为了获得这个精度,表达式中的字符和短整......