C语言动态内存的分配实例详解

目录
  • 前言
  • 动态内存分配的定义
  • 动态内存的优势
    • <1>  可以控制内存的大小
    • <2> 可以多次利用这部分空间
    • <3>不占用栈区的内存
  • malloc calloc realloc和free函数的介绍
  • 动态空间的申请与释放
    • 为单个对象分配空间
    •  为数组分配空间
  • 改变申请的动态内存(realloc的使用)
    • 扩容可能有三种情况
  • 总结

前言

给数组分配多大的空间?

你是否和初学C时的我一样,有过这样的疑问。

这一期博客就来聊一聊动态内存的分配

读完这篇文章,你可能对内存的分配有一个更好的理解

动态内存分配的定义

首先我们要搞清楚什么是动态内存的分配

平常我们定义的数组,都是在栈区分配的空间,都是分配的空间都是固定的大小

这种分配固定大小的内存分配方法称之为静态内存分配

与静态内存相对的,就是可以控制内存的分配的动态内存分配

注意:这里动态内存分配的空间是在堆区申请的,不是在栈区申请的

这里要讲一下什么是栈区,什么是堆区

内存的空间并不是都是一样的,在学习C语言时,提到的区域大致上分为栈区,堆区,和静态区。就比如说在一个车间一样,不同的区域做着不同的事,就有不同的功能,但是这些不同的功能又不是毫不相关的,他们彼此联系,相互构成整个内存空间.

动态内存的优势

<1>  可以控制内存的大小

在很多时候,我们申请的空间是未知的

就比如说通讯录,在刚刚开始用的时候很小的空间就足够了,但是在未来你不知道你需要存下多少个号码,这时候就存在一个问题,你定的空间需要多少个字节,当申请的太少,就会出现存不下去的情况,如果存的空间过大,有会造成一定的浪费。

在动态内存分配就可以避免这个问题,你可以运用 reallac 控制大小,当内存达到申请的空间时,就会主动扩容,也就是再次向内存申请空间。

<2> 可以多次利用这部分空间

静态内存分配利用的空间,整个程序结束才会释放给系统

而动态内存分配的空间,只能在函数运行结束后由系统自动释放,需要用户主动去释放,可以通过利用完(就比如说打印元素,打印完),用户再通过 free函数释放 这块申请的空间,当再次用动态内存申请空间时,就可以再次利用这块空间,这样也能在一定程度上,可以节省一定的空间。

<3>不占用栈区的内存

假设栈区定义了变量

而每个变量分配内存时,之间又有一定的间隙

当定义的变量足够多时,空隙也会很多

这时候向系统申请一个比较大且连续的空间时,虽然有足够的空间,但是缺少了连续的空间

就无法申请到这部分空间

所以动态内存在堆区申请,就完全不必担心栈区的空间不够的问题

说到这里,你是不是有一个疑惑,为什么空间的内存存在栈区和堆区之分

如果感兴趣,可以参考这个回答——为什么存在栈区堆区

malloc calloc realloc和free函数的介绍

在动态内存的分配中,离不开malloc与calloc,这两个函数都是向内存申请空间

 
                          calloc
头文件               #include <stdlib.h> 
格式                   void *calloc(size_t num, size_t size);
功能                   为num个大小为size字节的对象分配存储空间,该空间内的所有位都会初始化为。
返回值                若分配成功,则返回一个指向已分配的空间开头的指针;若分配失败,则返回空指针

这两个函数都是向系统申请动态内存空间,他们的头文件,返回值和功能大致都是相同的

不同的是calloc函数开辟的空间,就会将空间的内容全部初始话为零

而,malloc函数向系统申请的空间,空间的值都是随机的


realloc

头文件     #include <stdlib.h>
格式         void *realloc(void *mem_address, unsigned int newsize);
功能         先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将                            mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝                  到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不                    需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
返回值      如果重新分配成功则返回指向被分配内存的指针,否则返回空指针NULL。

动态空间的申请与释放

讲完动态内存申请的相关函数,那具体的代码实现是什么呢

<1>    double *x;

<2>    x=calloc(1,sizeof(double))或者x=malloc(sizeof(double));

<3>    free;

下面动态分配的内存赋值并显示

为单个对象分配空间

#include<stdio.h>
#include<stdlib.h>
int main()//动态内存的赋值与显示
{
	int* a;
	a = malloc(sizeof(int));    //分配动态内存
	if (a == NULL)              //是否成功分配了储存空间,否则返回分配失败
		printf("分配失败");
	else
	{
		*a = 20;
		printf("*a=%d\n", *a);
		free(a);                //释放
	}
	return 0;
}

 为数组分配空间

#include<stdio.h>
#include<stdlib.h>
int main()//动态内存的赋值与显示
{
	int n = 0;  int* a; int i = 0;
	printf("输入分配空间元素的个数:>");
	scanf_s("%d", &n);
	a =(int *) calloc(n,sizeof(int));
	if (a == NULL)
		printf("分配失败");
	else
	{
		for (i = 0; i < n; i++)
		{
			*(a + i) = i;
			printf("*a=%d\n", *(a+i));
		}
		free(a);
	}
	return 0;
}

这里其实没有“为数组开辟的空间”这一说

因为动态申请的空间都是一个一个的“

不难发现,calloc与malloc的差别并不大,只有第一个参数不同

在这两行代码中,存在着一个小细节

a = malloc(sizeof(int));

a =(int *) calloc(n,sizeof(int));

这两者的差别不仅仅是函数的不同,其中后者有强制类型转换,而前者没有

实际上在C语言的标准上,有无强制类型转换都是行得通的(当然在c++必须将强制类型转换)

因为无论是calloc还是malloc,他们的返回值都是void* ,这里的void*实际上可以转换为int*类型或者其他类型,换句话说,就是返回的指针是兼容所有类型的万能指针。

即指向void型的指针可以指向任意类型的对象,是一种特殊类型的指针。

指向void型的指针的值可以赋给指向任意类型的指针,反之亦可。

改变申请的动态内存(realloc的使用)

#include<stdlib.h>
#include<stdio.h>
int main()
{
	int* a; int i = 0;
	a=(int *)calloc(10,sizeof(int));
	if (a == NULL)
		printf("分配失败");
     //使用
	else
	{
		for (i = 0; i < 10; i++)
		{
			*(a + i) = i;
			printf("*a=%d\n", *(a + i));
		}
	//需要扩容
		int* ret = realloc(a, 80);
		if (ret != NULL)
		{
			a = ret;
		}
		free(a);
        a=NULL;
	}
}

扩容不能直接就a=realloc(a, 80),需要中间引一个中间变量*ret

扩容可能有三种情况

情况一(在a的地址处,有空余的空间来扩容)

情况二 (在a的地址处,没有空余的空间来扩容,但是有其他的空间可存储扩容后的空间)

情况三(reallo调整空间失败)

 在这三种情况中,第一种的地址不变

第二种会在一个新的地方申请足够大的地方,此时的地址不在是a原先的地址

第三种就扩容失败,就会导致扩容前申请的空间,也发生了改变,所以不能直接用a来重新赋值,

总结

到此这篇关于C语言动态内存分配的文章就介绍到这了,更多相关C语言动态内存分配内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言 动态内存分配详解

    C语言 动态内存分配详解 动态内存分配涉及到堆栈的概念:堆栈是两种数据结构.堆栈都是数据项按序排列的数据结构,只能在一端(称为栈顶(top))对数据项进行插入和删除. 栈(操作系统):由操作系统自动分配释放 ,存放函数的参数值,局部变量的值等.其操作方式类似于数据结构中的栈. 堆(操作系统): 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收,分配方式倒是类似于链表. \在C语言中,全局变量分配在内存中的静态存储区,非静态的局部变量(包括形参)是分配在内存的动态存储区,该存储区被

  • C语言中关于动态内存分配的详解

    目录 一.malloc 与free函数 二.calloc 三.realloc 四.常见的动态内存的错误 [C语言]动态内存分配 本期,我们将讲解malloc.calloc.realloc以及free函数. 这是个动态内存分配函数的头文件都是 <stdlib.h>. c语言中动态分配内存的函数,可能有些初学c语言的人不免要问了:我们为什么要通过函数来实现动态分配内存呢? 首先让我们熟悉一下计算机的内存吧!在计算机的系统中大致有这四个内存区域: 1)栈:在栈里面储存一些我们定义的局部变量以及形参(

  • C语言动态内存分配的详解

    C语言动态内存分配的详解 1.为什么使用动态内存分配 数组在使用的时候可能造成内存浪费,使用动态内存分配可以解决这个问题. 2. malloc和free C函数库提供了两个函数,malloc和free,分别用于执行动态内存分配和释放. (1)void *malloc(size_t size); malloc的参数就是需要分配的内存字节数.malloc分配一块连续的内存.如果操作系统无法向malloc提供更多的内存,malloc就返回一个NULL指针. (2)void free(void *poi

  • C语言编程C++动态内存分配示例讲解

    目录 动态内存管理 为什么存在动态内存分配 动态内存函数的介绍 malloc申请空间和free释放空间 有借有还 free释放内存 calloc申请内存 realloc调整动态内存的大小 realloc使用的注意事项 当然realloc也可以直接开辟空间 常见的动态内存错误 1.对NULL指针的解引用操作 2.对动态开辟空间的越界访问 3.对非动态开辟内存使用free释放 4.使用free释放一块动态内存开辟的一部分 5.对同一块动态内存多次释放 6.动态开辟内存忘记释放(内存泄漏) 几个面试题

  • c语言动态内存分配知识点及实例

    c语言怎么实现动态内存分配 我们经常会预先给程序开辟好内存空间,然后进行操作. int arr[5] ; 对这个数组我们在定义的时候必须给提前开辟好空间,并且在程序执行的过程中,这个开辟的内存空间是一直存在的,除非等到这个函数执行完毕,才会将空间释放.有个问题就是这个数组在程序中无法被修改. 这些问题给我们造成了一些使用上的不方便,所以,C中提供了malloc()函数. 关于malloc()函数,这个函数它接受一个参数:就是所需的内存的字节数.然后malloc()找到可用内存中那一个大小适合的块

  • C语言 动态内存分配的详解及实例

    1. 动态内存分配的意义 (1)C 语言中的一切操作都是基于内存的. (2)变量和数组都是内存的别名. ①内存分配由编译器在编译期间决定 ②定义数组的时候必须指定数组长度 ③数组长度是在编译期就必须确定的 (3)但是程序运行的过程中,可能需要使用一些额外的内存空间 2. malloc 和 free 函数 (1)malloc 和 free 用于执行动态内存分配的释放 (2)malloc 所分配的是一块连续的内存 (3)malloc 以字节为单位,并且返回值不带任何的类型信息:void* mallo

  • 详解C语言动态内存的分配

    目录 C 程序中,不同数据在内存中分配说明 内存动态分配的相关函数 应用实例 动态分配内存的基本原则 总结 C 程序中,不同数据在内存中分配说明 1)全局变量——内存中的静态存储区 2)非静态的局部变量——内存中的动态存储区——stack 栈 3)临时使用的数据—建立动态内存分配区域,需要时随时开辟,不需要时及时释放——heap 堆 4)根据需要向系统申请所需大小的空间,由于未在声明部分定义其为变量或者数组,不能通过变量名或者数组名 来引用这些数据,只能通过指针来引用) 内存动态分配的相关函数

  • C语言动态内存分配函数的实现

    在C中我们开辟内存空间有两种方式 : 1.静态开辟内存 :例如: int a;int b[10]; 这种开辟内存空间的特点是 所开辟的内存是在栈中开辟的固定大小的 ,如a是4字节 ,数组b是40字节 ,并且数组在申明时必须指定其长度 , 如果是全局数组的话,内存是在编译时分配好的,如果是局部变量数组的话,运行时在栈上静态分配内存.不管是全局数组还是局部数组,它们都有一个特点,那就是数组大小是确定的,是代码中写死的.那如果我们想在程序运行时才确定一个数组的大小 , 前两种在栈上分配内存的方法显然是

  • C语言编程动态内存分配常见错误全面分析

    目录 前言:为什么存在动态内存分配? 一.动态内存函数 1.malloc和free函数 2.calloc函数 3.realloc函数 二.常见错误 1.对NULL指针解引用 2.对动态开辟空间的越界访问 3.对非动态开辟使用free函数 4.使用free释放一块动态内存开辟内存的一部分 5.对同一块空间多次释放 6.动态开辟内存忘记释放 总结 前言:为什么存在动态内存分配? 我们已经掌握的内存开辟方式如下 int a=10;//在栈空间上开辟4字节 char arr[10]={0};//在栈空间

  • C语言动态内存的分配实例详解

    目录 前言 动态内存分配的定义 动态内存的优势 <1>  可以控制内存的大小 <2> 可以多次利用这部分空间 <3>不占用栈区的内存 malloc calloc realloc和free函数的介绍 动态空间的申请与释放 为单个对象分配空间  为数组分配空间 改变申请的动态内存(realloc的使用) 扩容可能有三种情况 总结 前言 给数组分配多大的空间? 你是否和初学C时的我一样,有过这样的疑问. 这一期博客就来聊一聊动态内存的分配 读完这篇文章,你可能对内存的分配有一

  • Kotlin 语言中调用 JavaScript 方法实例详解

    Kotlin 语言中调用 JavaScript 方法实例详解 Kotlin 已被设计为能够与 Java 平台轻松互操作.它将 Java 类视为 Kotlin 类,并且 Java 也将 Kotlin 类视为 Java 类.但是,JavaScript 是一种动态类型语言,这意味着它不会在编译期检查类型.你可以通过动态类型在 Kotlin 中自由地与 JavaScript 交流,但是如果你想要 Kotlin 类型系统的全部威力 ,你可以为 JavaScript 库创建 Kotlin 头文件. 内联 J

  • C语言实现“幸运数”的实例详解

    C语言实现"幸运数"的实例详解 1.题目: 标题:幸运数 幸运数是波兰数学家乌拉姆命名的.它采用与生成素数类似的"筛法"生成. 首先从1开始写出自然数1,2,3,4,5,6,-. 1 就是第一个幸运数. 我们从2这个数开始.把所有序号能被2整除的项删除,变为: 1 _ 3 _ 5 _ 7 _ 9 -. 把它们缩紧,重新记序,为: 1 3 5 7 9 -. .这时,3为第2个幸运数,然后把所有能被3整除的序号位置的数删去.注意,是序号位置,不是那个数本身能否被3整除!

  • C语言数据输入与输出实例详解

    C语言数据输入与输出实例详解 1 概论 C语言提供了跨平台的数据输入输出函数scanf()和printf()函数,它们可以按照指定的格式来解析常见的数据类型,例如整数,浮点数,字符和字符串等等.数据输入的来源可以是文件,控制台以及网络,而输出的终端可以是控制台,文件甚至是网页. 2 数据输出 从第一个c语言程序中,就使用了跨平台的库函数printf实现将一段文字输出到控制台,而实际上,printf()不仅可以将数据按照指定的格式输出到控制台,还可以是网页或者是指定的文件中,printf()函数执

  • C语言线性表顺序存储结构实例详解

    C语言线性表顺序存储结构实例详解 1. 什么是顺序存储结构? 用一段地址连续的存储单元依次存储线性表的数据元素. 2.线性表的顺序存储结构 #include<stdio.h> #include<stdlib.h> #define Max 80 //存储空间初始分配量 #define Increment 10 //存储空间分配增量 typedef struct { int *elem; // 存储空间基地址,此处为int型,视情况而定 int length; // 元素表当前长度 i

  • C语言树状数组的实例详解

    C语言树状数组的实例详解 最近学了树状数组,给我的感觉就是 这个数据结构好神奇啊^_^ 首先她的常数比线段树小,其次她的实现复杂度也远低于线段树 (并没有黑线段树的意思=-=) 所以熟练掌握她是非常有必要的.. 关于树状数组的基础知识与原理网上一搜一大堆,我就不赘述了,就谈一些树状数组的应用好了 1,单点修改,求区间和 #define lowbit(x) (x&-x) // 设 x 的末尾零的个数为 y , 则 lowbit(x) == 2^y void Update(int i,int v)

  • Spring AOP 动态多数据源的实例详解

     Spring AOP 动态多数据源的实例详解 当项目中使用到读写分离的时候,我们就会遇到多数据源的问题.多数据源让人最头痛的,不是配置多个数据源,而是如何能灵活动态的切换数据源.例如在一个spring和Mybatis的框架的项目中,我们在spring配置中往往是配置一个dataSource来连接数据库,然后绑定给sessionFactory,在dao层代码中再指定sessionFactory来进行数据库操作. 正如上图所示,每一块都是指定绑死的,如果是多个数据源,也只能是下图中那种方式. 可看

  • C语言中联合体union的实例详解

     C语言中联合体union的实例详解 1.定义: union(int i, short s, char c) un; un.i = 3; printf("i=%d",un.i); printf("length = %d\n",sizeof(un);//==4,有最大的变量来决定 2.相当与java里的List T类型 3.数据交换 void swap(int *p , int *q){ int temp = *p; *p = *q; *q = temp; } 4.打

  • C语言中二级指针的实例详解

    C语言中二级指针的实例详解 用图说明 示例代码: #include <stdio.h> int main(int argc, const char * argv[]) { // int a = 5; int *p1 = &a; //-打印地址-----地址相同--------------- printf("&a = %p\n", &a);// printf("p1 = %p\n", p1);// int **p2 = &p

  • C语言中调用Swift函数实例详解

    C语言中调用Swift函数实例详解 在Apple官方的<Using Swift with Cocoa and Objectgive-C>一书中详细地介绍了如何在Objective-C中使用Swift的类以及如何在Swift中使用Objective-C中的类.在后半部分也介绍了如何在Swift中使用C函数,不过对于如何在C语言中使用Swift函数却只字未提.这里我就为大家分享一下如何在C语言中调用Swift函数. 我们首先要知道的是,所有Swift函数都属于闭包.其次,Swift函数的调用约定与

随机推荐