C语言 动态内存管理全面解析

 更新时间:2022年02月11日 10:01:15   作者:i跑跑  
动态内存是相对静态内存而言的。所谓动态和静态就是指内存的分配方式。动态内存是指在堆上分配的内存,而静态内存是指在栈上分配的内存,本文带你深入探究C语言中动态内存的管理

1. 为什么存在动态内存分配

*动态内存开辟在堆区*

我们已经掌握的开辟内存方式是类型直接定义变量,开辟的内存是固定的,像:

int a=20;  //在栈空间上开辟四个字节

还有数组,我们可以指定开辟空间的大小,像:

char arr[10] = {0};  ///在栈空间上开辟10个字节的连续空间

但在程序运行时,很多时候我们会遇到内存不够或者内存过多引起的浪费问题,那么有没有那种使用多少内存开辟多少内存的方法?这就是本篇文章要介绍的动态内存。

2. 动态内存函数的介绍

2.1 malloc和free

malloc和free都声明在 stdlib.h 头文件中

void* malloc (size_t size);
//向内存申请一块连续可用的空间,并返回指向这块空间的指针

如果开辟成功,则返回一个指向开辟好空间的指针。 如果开辟失败,则返回一个NULL指针。 返回值的类型是 void* ,malloc函数并不知道开辟空间的类型,在使用的时候自己来决定。

void free (void* ptr);
//free函数用来释放动态开辟的内存

如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。

如果参数 ptr 是NULL指针,则函数什么事都不做。

举例:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
int main()
{
	//开辟10个整型空间
	int* p = (int*)malloc(40);
	if (NULL==p)
	{
		printf("%s\n",strerror(errno));  //判断开辟失败的原因
		return 0;
	}
	//使用
 
	//释放
	free(p);  //将空间还给系统,但是里面的内容没有改变,还可以通过p来找到地址
	p = NULL;  //因此要将地址置为空指针
	return 0;
}

2.2 calloc 

void* calloc (size_t num, size_t size);
//num为元素个数,size为每个元素的大小

注:与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

当用calloc来开辟10个整型空间时

int* p = (int*)calloc(10,sizeof(int));

2.3 realloc

void* realloc (void* ptr, size_t size);
//ptr 是要调整的内存地址     size 调整之后新大小
//返回值为调整之后的内存起始位置

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

1.当原地址后有足够的空间时,可以接着原地址连续开辟空间,最后返回起始地址。

2.当原地址后空间不足以开辟我们所需要的空间时,那么realloc会自动寻找一块足以存放我们需要的的空间,并将原地址的内容复制到新空间中,释放掉原地址中的内容,返回开辟出空间的初始地址。

我们可以先判断是否开辟成功,再将地址赋予p

3. 常见的动态内存错误

3.1 对NULL指针的解引用操作

开辟动态内存时,一定要注意对返回空指针的函数要进行判断,防止对空指针进行解引用,以免程序出现问题。

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

 int *p = (int *)malloc(10*sizeof(int));   //开辟内存
 
 if(NULL == p)    //判断是否开辟成功
 {
 exit(EXIT_FAILURE);
 }
 
 int i=0;
 for(i=0; i<=10; i++)
 {
 *(p+i) = i;//当i是10的时候越界访问
 }
 free(p);

这块可以像理解数组一样,不能访问下标为10的地址,会造成越界访问。

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

void test()
{
	int a = 10;
	int *p = &a;
	free(p);
}
 
 
int main()
{
	test();
	return 0;
}

不是动态内存开辟的空间内存不在堆区,没必要用free释放,在栈区开辟的空间在出了作用域后会自动还给系统,没有必要,也不允许用free进行释放。

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

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

 不支持释放一部分内存,这样的写法不支持不可取。只能从动态内存开辟的起始位置来进行释放。

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

void test()
{
 int *p = (int *)malloc(100);
 free(p);
 free(p);//重复释放
}

重复释放也会报错

当p第一次释放后,将p=NULL,再次释放的话就不会有问题;写代码是要避免重复释放的情况,同时要记住每次释放完之后都要将地址置为空指针。

若忘记释放开辟的空间,就会造成内存泄漏的问题(在释放该段内存之前就失去了对该段内存的控制,从而造成了内存的浪费)

到此这篇关于C语言 动态内存管理深入理解的文章就介绍到这了,更多相关C语言 动态内存管理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Qt编写地图实现实时动态轨迹效果

    Qt编写地图实现实时动态轨迹效果

    实时动态轨迹主要是需要在地图上动态显示GPS的运动轨迹,也是编写地图时一个重要的功能。本文将利用Qt实现这一功能,需要的可以参考一下
    2022-02-02
  • C++学校运动会管理系统的实现

    C++学校运动会管理系统的实现

    这篇文章主要为大家详细介绍了C++如何实现学校运动会管理系统,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-10-10
  • C语言获取文件大小的两种方式

    C语言获取文件大小的两种方式

    因为音视频开发的需要,经常会写一些文件输入输出的测试程序,常常用到获取文件大小的函数,本篇文章就记录一下常用的两种获取文件大小的方式,希望对大家有所帮助
    2023-11-11
  • C++中的vector容器对象学习笔记

    C++中的vector容器对象学习笔记

    这篇文章主要介绍了C++中的vector容器对象学习笔记,其中文章最后标红的resize与reserve方法的差别特别需要注意,需要的朋友可以参考下
    2016-05-05
  • C++常用字符串函数大全(2)

    C++常用字符串函数大全(2)

    这篇文章主要给大家分享的是C++常用字符串函数的大全,cstring.h库即C语言中的string.h库,它是C语言中为字符串提供的标准库。C++对此进行了兼容,所以我们在C++当中一样可以使用,下面文章的详细内容,需要的朋友可以参考一下
    2021-11-11
  • 贪心算法的C语言实现与运用详解

    贪心算法的C语言实现与运用详解

    这篇文章主要介绍了贪心算法的C语言实现与运用详解,运用么,就是文中所附的ACM练习题,哈哈:D需要的朋友可以参考下
    2015-08-08
  • C经典算法之二分查找法

    C经典算法之二分查找法

    这篇文章主要介绍了C经典算法之二分查找法的相关资料,这里提供两种方法帮助大家实现这样的功能,需要的朋友可以参考下
    2017-10-10
  • C++ Vector用法深入剖析

    C++ Vector用法深入剖析

    C++ Vector应用方式是比较特殊的,我们将会在这篇文章中针对于它的应用方式进行一个详细的介绍,希望大家能充分掌握这一应用技巧
    2014-08-08
  • 关于C++中push_back()函数的用法及代码实例

    关于C++中push_back()函数的用法及代码实例

    push_back是vector的一个方法,表示将一个元素存储到容器的末尾,下面这篇文章主要给大家介绍了关于C++中push_back()函数用法的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • C语言中的运算符优先级和结合性一览表

    C语言中的运算符优先级和结合性一览表

    这篇文章主要介绍了C语言中的运算符优先级和结合性一览表,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02

最新评论