C语言的动态内存管理的深入了解

 更新时间:2022年02月08日 17:02:23   作者:太阳风暴  
这篇文章主要为大家详细介绍了语言C的动态内存管理,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

一、动态内存分配

  • (1)用malloc类的函数分配内存;
  • (2)用这些内存支持应用程序;
  • (3)用free函数释放内存。

内存的简答来说的三大操作:分配----使用----释放

内存管理指的是:分配— ----释放

我们编写的程序代码:使用

程序本质上就是处理数据,数据信息需要存放在内存里,就是用二极管表示的开断表示二进制数,进一步用二进制数表示万物:如音乐、文字、视频、图片、等等各种资源。

分配–释放:为了更好的利用和回收内存资源,最大程度的发挥计算资源,统一操作系统来调度

动态内存分配其实就要谈到自动内存分配

自动分配:内存其实就是我们在编程文件里定义的变量实际在运行时映射的内存,这部分变量的内存都是由系统自动管理(分配-释放)。函数体(或者语句块)内的变量都是分布在函数  里面,运行时就会把这个帧插入到函数运行栈里,内存这些都由系统分拨,运行完就出栈,内存也被操作系统回收,一致到主函数出栈,程序退出。

动态分配:就是我们自己手动申请操作系统给我们程序分配的内存,内存区域主要在于  上,这部分资源是我们手动申请和回收的。分配到的资源是我们来操作、存放数据的地方。

实际上我们定义的变量最后也会被翻译为地址,都是通过寻址来操作变量的值(可以去看看汇编语言)

#include <stdio.h>
#include <stdlib.h>
 int main(void)
{
	int*pi=(int*)malloc(sizeof(int));
	*pi=5;
	printf("*pi:%d\n",*pi);
	free(pi);
 	return 0;
}

在这里插入图片描述

注意点

int *pi=(int)malloc((4));

  • 然而,依赖于系统所用的内存模型,整数的长度可能会发生变化。可移植的方法是使用sizeof操作符,这样不管程序在哪里运行都会返回正确的长度。
  • 使用(int)malloc(number * (sizeof(int)));*

二、动态内存分配函数

有几个内存分配函数可以用来管理动态内存,虽然具体可用的函数取决于系统,但大部分系统的stdlib.h头文件中都有如下函数:

malloc()

realloc()

calloc()

函数描述
malloc从堆上分配内存
realloc在之前分配的内存块的基础上,将内存重新分配为更大或者更小的部分
calloc从堆上分配内存并清零

1、malloc()

malloc函数从堆上分配一块内存,所分配的字节数由该函数唯一的参数指定,返回值是void指针,如果内存不足,就会返回NULL。此函数不会清空或者修改内存。

声明:voidmalloc(size_t);

  • (1)从堆上分配内存;
  • (2)内存不会被修改或是清空;
  • (3)返回首字节的地址。

实例用法int* pi=(int*)malloc(sizeof(int));

因为当malloc无法分配内存时会返回NULL,在使用它返回的指针之前先检查NULL是不错的做法,如下所示:

int*pi=(int*)malloc(sizeof(int));
if(pi!=NULL)
{
	//指针没有问题
}else
{
	//无效的指针
}
  • (4)静态、全局指针和malloc

初始化静态或全局变量时不能调用函数。下面的代码声明一个静态变量,并试图用malloc来初始化:

*static int pi = malloc(sizeof(int));

这样会产生一个编译时错误消息,全局变量也一样。

对于静态变量,可以通过在后面用一个单独的语句给变量分配内存来避免这个问题。但是全局变量不能用单独的赋值语句,因为全局变量是在函数和可执行代码外部声明的,赋值语句这类代码必须出现在函数中:

static int *pi;
pi = malloc(sizeof(int)); 

2、realloc()

声明:void *realloc(void *ptr,size t size);

realloc函数返回指向内存块的指针。该函数接受两个参数,第一个参数是指向原内存块的指针,第二个是请求的大小。重新分配的块大小和第一个参数引用的块大小不同。返回值是指向重新分配的内存的指针。

第一个参数第二个参数行为
同malloc
非空0原内存块被释放
非空比原内存块小利用当前的块分配更小的块
非空比原内存块大要么在当前位置要么在其他位置分配更大的块

在这里插入图片描述

堆管理器可以重用原始的内存块,且不会修改其内容。不过程序继续使用的内存超过了所请求的8字节。也就是说,我们没有修改字符串以便它能装进8字节的内存块中。在本例中,我们本应该调整字符串的长度以使它能装进重新分配的8字节。实现这一点最简单的办法是将NUL赋给地址507。实际使用的内存超出分配的内存不是个好做法,应该避免。

在这里插入图片描述

3、calloc()

calloc函数在申请内存时会清空内存【清空内存的意思是将其内容置为二进制0】

声明: void *calloc(size_t numElements,size_t elementSize);

calloc函数会根据numElements和elementSize两个参数的乘积来分配内存,并返回一个指向内存的第一个字节的指针。如果不能分配内存,则会返回NULL。此函数最初用来辅助分配数组内存。

如果numElements或elementSize为0,那么calloc可能返回空指针。如果calloc无法分配内存就会返回空指针,而且全局变量errno会设置为ENOMEM(内存不足),这是POSIX错误码,有的系统上可能没有。

三、用free函数释放内存

有了动态内存分配,程序员可以将不再使用的内存返还给系统,这样可以释放内存
留作他用。通常用free函数实现这一点,该函数的原型如下:

声明: void free(void *ptr);

指针参数应该指向由malloc类函数分配的内存的地址,这块内存会被返还给堆。尽管指针仍然指向这块区域,但是我们应该将它看成指向垃圾数据。稍后可能重新分配这块区域,并将其装进不同的数据。

在这里插入图片描述

要点

  • 释放含义:指的是释放堆上的申请内存,其实就是告诉堆管理器,这个资源我不用了,可以回收了
  • 本地还是保留了之前申请内存的地址,这个地址我们应该避免去使用,也就是置这个指针为NULL
  • 不能再去接引已释放资源指针的值
  • 不能重复多次释放指针指向的内存(free)

四、迷途指针

如果内存已经释放,而指针还在引用原始内存,这样的指针就称为迷途指针。迷途指针没有指向有效对象,有时候也称为过早释放
迷途指针带来的问题:

  • 如果访问内存,则行为不可预期;
  • 如果内存不可访问,则是段错误;
  • 潜在的安全隐患。

造成的原因:

  • 访问已释放的内存;
  • 返回的指针指向的是上次函数调用中的自动变量;
//第一种情况
int*pi = (int*)malloc(sizeof(int));
printf("*pi:%d\n",*pi);
free(pi);
*pi = 10;
 //第二种情况
int*p1 = (int*)malloc(sizeof(int));
*p1 = 5;
int* p2;
p2 = p1;
free(p1);
*p2 = 10;//迷途指针
 //第三种情况
/*
大部分编译器都把块语句当做一个栈帧。tmp变量分配在栈帧上,之后在块语句退出时会出栈。
pi指针现在指向一块最终可能被其他活跃记录(比如foo函数)覆盖的内存区域。
图2-13说明的就是这种情形。
*/
int *pi;
int tmp = 5;
pi = &tmp;
//这里pi变成了迷途指针
foo();

在这里插入图片描述

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!      

相关文章

  • QT开发应用程序的欢迎界面实例

    QT开发应用程序的欢迎界面实例

    下面小编就为大家带来一篇QT开发应用程序的欢迎界面实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • C++ CryptoPP使用AES实现加解密详解

    C++ CryptoPP使用AES实现加解密详解

    Crypto++ (CryptoPP) 是一个用于密码学和加密的 C++ 库,提供了大量的密码学算法和功能,这篇文章主要为大家介绍了C++ CryptoPP如何使用AES实现加解密,需要的可以参考下
    2023-11-11
  • C++中基本的输入输出函数使用指南

    C++中基本的输入输出函数使用指南

    这篇文章主要介绍了C++中基本的输入输出函数使用指南,是C++入门学习中的基础知识,需要的朋友可以参考下
    2015-09-09
  • 详解C++实现链表的排序算法

    详解C++实现链表的排序算法

    链表排序思想和数组排序类似,区别就是数组遍历容易,数据交换也容易;链表(单项链表)只能一个方向遍历,不能逆序遍历,且不能随机访问,所以排序比较麻烦。本文将详细介绍链表排序的方式,并且用C++来实现
    2021-06-06
  • VScode上配置 c语言环境的图文教程

    VScode上配置 c语言环境的图文教程

    这篇文章主要介绍了配置VScode c语言环境,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • c++网络编程下Linux的epoll技术和Windows下的IOCP模型

    c++网络编程下Linux的epoll技术和Windows下的IOCP模型

    c++ 网络编程LINUX-epoll/windows-IOCP下socket opoll函数用法 优于select方法的epoll 以及windows下IOCP 解决多进程服务端创建进程资源浪费问题,感兴趣的小伙伴一起来学习吧
    2021-08-08
  • C++超详细讲解标准库

    C++超详细讲解标准库

    C++强大的功能来源于其丰富的类库及库函数资源。C++标准库(C++ Standard Library, 亦可称作,C++标准程序库)的内容总共在50个标准头文件中定义。在C++开发中,要尽可能地利用标准库完成
    2022-06-06
  • 一文掌握scanf的用法实例小结

    一文掌握scanf的用法实例小结

    scanf的基本用法除了常规的输入操作外还有一些特殊的用法,使用这些用法可以很方便的在输入中读取想要的数据,这篇文章主要介绍了scanf的用法,需要的朋友可以参考下
    2023-12-12
  • Linux下semop等待信号时出现Interrupted System Call错误(EINTR)解决方法

    Linux下semop等待信号时出现Interrupted System Call错误(EINTR)解决方法

    本篇文章是对在Linux下semop等待信号时出现Interrupted System Call错误(EINTR)的解决方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++实例讲解四种类型转换的使用

    C++实例讲解四种类型转换的使用

    在C++语言中新增了四个关键字static_cast、const_cast、reinterpret_cast和dynamic_cast。这四个关键字都是用于类型转换的,类型转换(type cast),是高级语言的一个基本语法。它被实现为一个特殊的运算符,以小括号内加上类型名来表示,接下来让我们一起来详细了解
    2022-06-06

最新评论