IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C语言动态内存管理(malloc,calloc,free,realloc) -> 正文阅读

[C++知识库]C语言动态内存管理(malloc,calloc,free,realloc)

前言

为什么会引出动态内存开辟这个功能呢?

那是因为向我们之前学习的数组,一旦创建之后所占用的空间就不会改变,可能会造成空间的浪费或者开辟空间的不足。

一,malloc

void* malloc(size_t num)
1,函数的返回类型为空指针类型,如果为开辟成功那么返回NULL
2,函数的参数为开辟空间的大小(单位:字节)
3,函数开辟的空间十载内存中的堆区开辟的,会在整个程序结束后自动销毁
4,演示

#include<stdio.h>
#include<stdlib.h>
int main()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	if (p == NULL)
	{
		perror("malloc");
	}
	else
	{
		int i = 0;
		for (i = 0; i < 5; i++)
		{
			*(p + i) = i;
		}
		for (i = 0; i < 5; i++)
		{
			printf("%d ", *(p + i));
		}
	}
	free(p);
	p = NULL;
	return 0;
}

在这里插入图片描述
注意 :开辟不成功返回NULL

#include<stdio.h>
#include<stdlib.h>
#include<limits.h>
int main()
{
	int* p = (int*)malloc(INT_MAX*sizeof(int));
	if (p == NULL)
	{
		perror("malloc");
	}
	else
	{
		int i = 0;
		for (i = 0; i < 5; i++)
		{
			*(p + i) = i;
		}
		for (i = 0; i < 5; i++)
		{
			printf("%d ", *(p + i));
		}
	}
	free(p);
	p = NULL;
	return 0;
}

在这里插入图片描述

二,free

void free(void* ptr)
1,函数的返回值为空,函数的参数为空指针
2,作用是将ptr指向的动态开辟的空间释放
3,可以free(NULL)
4,我们在动态开辟内存空间后,并且使用完后,要及时free释放空间,避免造成内存泄漏。
5,free掉空间后,最好将原指针指向NULL,避免造成非法访问
在这里插入图片描述

三,calloc

void* calloc(size_t num,size_t n)
1,函数的返回值为空指针类型
2,第一个参数表示开辟元素的个数,第二个参数表示每个元素的大小(单位:字节)
3,calloc与malloc大体相同,区别为calloc申请的空间会被初始化为0.
在这里插入图片描述
在这里插入图片描述

四,realloc

void* realloc (void* ptr, size_t size)
1,函数的返回值为空指针类型
2,第一个参数为空指针类型是一块动态开辟空间的地址,第二个参数是重新调整后动态开辟空间的大小
3,realloc(NULL,40)==malloc(NULL,40)
4,这个函数可以对之前开辟的空间重新调整可大可小
5,在用realloc函数扩容的时候,若原空间后面的空间充足,那么新开辟的空间就是与原空间挨着的后面的空间
在这里插入图片描述
6,在用realloc函数扩容的时候,若原空间后面的空间不充足,那么会找到一块新的空间大小为扩容后的大小,并且会将原空间的内容拷贝到新空间中,还会释放掉原空间。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
可以看到当扩容的空间较小时,会直接在原空间的基础上开辟新的空间。

在这里插入图片描述

在这里插入图片描述

可以看到当扩容的空间较大时,系统会重新找到一块空间。

五,动态内存管理的常见问题

1,对空指针进行解引用操作

void test2()
{
	int* p = (int*)malloc(sizeof(int) * INT_MAX);
	*p = 20;
	free(p);
}

这里如果p为空指针的话,那么就会对空指针解引用。

如何改进呢?

void test2()
{
	int* p = (int*)malloc(sizeof(int) * INT_MAX);
	if (p == NULL)
	{
		perror("malloc");
	}
	else
	{
		*p = 20;
	}
	free(p);
	p = NULL;
}

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

void test3()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	if (p != NULL)
	{
		int i = 0;
		for (i = 0; i < 10; i++)
		{
			*(p + i) = i;
		}
	}
	free(p);
	p = NULL;
}

这里看到我们用malloc开辟了20个字节的内容,而我们在赋值的时候却访问了40个字节的内容,这就造成了,对动态开辟内存空间的越界访问。

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

void test4()
{
	int a = 10;
	free(&a);
}

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

void test5()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	if (p != NULL)
	{
		int i = 0;
		for (i = 0; i < 5; i++)
		{
			*p = i;
			p++;
		}
	}
	free(p);
	p = NULL;
}

我们在赋值的时候已经将p指向的空间移动,这样之后就不能再用free(p)来释放动态开辟的空间

5,对一块动态开辟的空间多次释放

void test6()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	free(p);
	//...
	//...
	//...
	//...
	free(p);
}

6,动态开辟的内存空间忘记释放(内存泄漏)

这就是内存开辟之后,使用完忘记释放空间。

void test5()
{
	int* p = (int*)malloc(sizeof(int) * 5);
	if (p != NULL)
	{
		int i = 0;
		for (i = 0; i < 5; i++)
		{
			*(p + i) = i;
		}
	}
	
}

:以上就是动态内存开辟过程中容易出错的六个地方,在你写代码的时候要注意哟。

六,四道经典面试题目

每道题目哪里出了问题?

第一道

void GetMemory(char *p)
{
p = (char *)malloc(100);
}
void Test(void)
{
char *str = NULL;
GetMemory(str);
strcpy(str, "hello world");
printf(str);
}

1,对p开辟的空间,并不会带回函数test中,所以str一直指向的都是NULL,在调用strcpy时,会造成对空指针解引用。
2,对动态开辟的空间没有free

纠正

void GetMemory(char** p)
{
	*p = (char*)malloc(100);
}
void Test(void)
{
	char* str = NULL;
	GetMemory(&str);
	if (str != NULL)
	{
		strcpy(str, "hello world");
		printf(str);
	}
	free(str);
	str = NULL;
}

第二道

char *GetMemory(void)
{
char p[] = "hello world";
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}

这是一种常见的错误就是,返回栈空间地址,函数返回了p指针p但是指针p指向的内容已被销毁。

第三道

void GetMemory(char **p, int num)
{
*p = (char *)malloc(num);
}
void Test(void)
{
char *str = NULL;
GetMemory(&str, 100);
strcpy(str, "hello");
printf(str);
}

1,主要问题是没有释放动态开辟的内存空间造成内存泄漏。
2,还有一个小问题就是在使用str时没有提前判断str是否为空指针

纠正

void GetMemory2(char** p, int num)
{
	*p = (char*)malloc(num);
}
void Test2(void)
{
	char* str = NULL;
	GetMemory(&str, 100);
	if (str != NULL)
	{
		strcpy(str, "hello");
		printf(str);
	}
	free(str);
	str = NULL;
}

第四道

void Test(void)
{
char *str = (char *) malloc(100);
strcpy(str, "hello");
free(str);
if(str != NULL)
{
strcpy(str, "world");
printf(str);
}
}

1,这道题的问题所在是,释放掉str所指向的空间后,没有将str指向空指针。这造成了调用strcpy函数时形成了非法访问。

纠正

void Test3(void)
{
	char* str = (char*)malloc(100);
	strcpy(str, "hello");
	free(str);
	str = NULL;
	if (str != NULL)
	{
		strcpy(str, "world");
		printf(str);
	}
}

以上四道题目都出自《高质量的C/C++编程》这本书中。

链接:https://pan.baidu.com/s/19mSw3miWFz7GYaMfaZ3uIw
提取码:kbtz

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-09-25 23:05:06  更:2022-09-25 23:06:31 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年4日历 -2024/4/17 7:33:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码