C语言进阶--动态内存管理(一)

目录

一.动态内存存在的原因

1.当前掌握的2种申请内存的方式

二. 动态内存函数

1. malloc()函数

(1)参数和返回类型

(2)函数功能

(3)内存释放 

(4)函数使用 

2.free()函数 

(1)参数和返回类型

(2)函数功能 

(3)函数应用

3.calloc()函数 

(1)参数和返回类型 

(2)函数功能 

(3)函数应用 

4.realloc()函数  

(1)参数和返回类型

(2)函数功能

(3)函数使用 

三.常见的动态内存错误

1.对空指针的解引用操作 

2.对动态开辟空间的越界使用

3.对非动态开辟内存使用free释放 

4.使用free释放一块动态开辟内存的一部分

5.对同一块动态内存多次释放

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


一.动态内存存在的原因

1.当前掌握的2种申请内存的方式

int main()
{
	int val = 20;//在栈空间上开辟四个字节,存放一个值
	char arr[10] = { 0 };//在栈空间上开辟10个字节的连续空间,存放一组数

	return 0;
}
  • 特点

1. 空间开辟大小是固定的。
2. 数组在声明的时候,必须指定数组的长度,它所需要的内存在编译时分配。 

二. 动态内存函数

1. malloc()函数

(1)参数和返回类型

void* malloc (size_t size)
  • 头文件 
<stdlib.h>

(2)函数功能

这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。

  • 如果开辟成功,则返回一个指向开辟好空间的指针,指向申请到的空间的起始地址。
  • 如果开辟失败(一般为所需开辟的空间过大),则返回一个NULL指针,因此malloc的返回值一定要做检查
  • 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
  • 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。 

(3)内存释放 

malloc函数申请的空间有两种释放方式:

  • 调用free函数释放--主动
  • 程序退出后,申请的空间被操作系统回收 --被动

(4)函数使用 

int main()
{
	int*p=(int*)malloc(10 * sizeof(int));//申请一块空间用来存放10个整型
	//判断是否开辟成功
	if (p == NULL)
	{
		perror("malloc");
		return 1;
	}
	//使用
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*(p + i) = i;
	}
	for (i = 0; i < 10; i++)
	{
		printf("%d ", p[i]);
	}
	return 0;
}

输出结果:

2.free()函数 

(1)参数和返回类型

void free (void* ptr);//ptr为要释放空间的起始地址
  •  头文件
<stdlib.h>

(2)函数功能 

free函数用来释放动态开辟的内存。

  • 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
  • 如果参数 ptr 是NULL指针,则函数什么事都不做 
  • free函数只能释放动态开辟的空间,非动态开辟的空间不能释放
  •  图示解析

(3)函数应用

int main()
{
	int*p=(int*)malloc(10 * sizeof(int));//申请一块空间用来存放10个整型
	//判断是否开辟成功
	if (p == NULL)
	{
		perror("malloc");
		return 1;
	}
	//使用
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		*(p + i) = i;
	}
	for (i = 0; i < 10; i++)
	{
		printf("%d ", p[i]);
	}

	free(p);
	p = NULL;
	return 0;
}

3.calloc()函数 

(1)参数和返回类型 

void* calloc (size_t num, size_t size)
  • 头文件 
<stdlib.h>

(2)函数功能 

  • 函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。
  • 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。

(3)函数应用 

int main()
{
	int* p =(int*)calloc(10, sizeof(int));
	if (p == NULL)
	{
		perror("calloc");
		return 1;
	}
	int i = 0;
	for (i = 0; i < 10; i++)
	{
		printf("%d ", p[i]);
	}
	free(p);
	p = NULL;
	return 0;
}

 结果:

4.realloc()函数  

(1)参数和返回类型

void* realloc (void* ptr, size_t size);
  •  头文件
<stdlib.h>

(2)函数功能

  • ptr 是要调整的内存地址
  • size 调整之后新大小
  • 返回值为调整之后的内存起始位置。 
  • 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间

realloc在调整内存空间的是存在两种情况:

  • 情况1:原有空间之后有足够大的空间

当是情况1 的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化

  • 情况2:原有空间之后没有足够大的空间

当是情况2 的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适大小的连续空间来使用。这样函数返回的是一个新的内存地址

注:realloc开辟空间也可能失败,失败的时候返回NULL

(3)函数使用 

int main()
{
	    int* p = (int*)calloc(10, sizeof(int));
		if (p == NULL)
		{
			perror("calloc");
			return 1;
		}
		int i = 0;
		for (i = 0; i < 10; i++)
		{
			*(p + i) = i;
		}
		for (i = 0; i < 10; i++)
		{
			printf("%d ", p[i]);
		}
		//空间不够,希望调整空间为20个整型的空间
		//p=realloc(p, 20 * sizeof(int));尽量避免这种写法,以防开辟新的空间失败,p成为空指针

		int* p2 = (int*)realloc(p, 20 * sizeof(int));
		if (p2 != NULL)
		{
			p = p2;
		}

		//释放
		free(p);
		p = NULL;
	    return 0;
}
  • 图示解析

 三.常见的动态内存错误

1.对空指针的解引用操作 

void test()
{
	int* p = (int*)malloc(40);
	*p = 20;//如果p的值是NULL,就会有问题
	free(p);
}

 2.对动态开辟空间的越界使用

void test()
{
	int* p = (int*)malloc(10 * sizeof(int));
	if (NULL == p)
	{
		perror("malloc");
		return 1;
	}
	int i = 0;
	for (i = 0; i <= 10; i++)
	{
		*(p + i) = i;//当i是10的时候越界访问
	}
	free(p);
}

3.对非动态开辟内存使用free释放 

void test()
{
  int a = 10;
  int *p = &a;
  free(p);//发生错误
  p=NULL;
}

4.使用free释放一块动态开辟内存的一部分

void test()
{
  int *p = (int *)malloc(100);
  p++;
  free(p);//p不再指向动态内存的起始位置
  p=NULL;
}

5.对同一块动态内存多次释放

void test()
{
  int *p = (int *)malloc(100);
  free(p);
  free(p);//重复释放
}
  • 解决方案 
void test()
{
  int *p = (int *)malloc(100);
  free(p);
  p=NULL;
  free(p);//这样重复释放不会报错
}

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

void test()
{
	int* p = (int*)malloc(100);
	if (NULL != p)
	{
		*p = 20;
		return;//这里return之后,后面的free语句就不执行了
	}

	free(p);
	p = NULL;
}
//出了函数以后指针变量p被回收,不仅找不到申请内存空间的地址,而且没有机会释放那块空间了
int main()
{
	test();
	//没机会释放已经申请的空间了,如果申请的空间很多,会导致内存耗完
	while (1);
	return 0;
}
  •  改善写法
int * test()
{
	int* p = (int*)malloc(100);
	if (NULL != p)
	{
		*p = 20;
		return p;//这里return之后,后面的free语句就不执行了
	}

	free(p);
	p = NULL;
}
int main()
{
	int*ret=test();
	free(ret);
	
	while (1);
	return 0;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值