C基础-对malloc的使用与理解
一、malloc函数分析
1.函数原型
void * malloc(size_t size);
2.Function(功能)
Allocates a block of size bytes of memory, returning a pointer to the beginning of the block
译:在内存中分配 size 个字节的空间,返回一个指向”内存空间开头“的指针(指针类型为void *)。
3.Parameters
size
size of the memory block,in bytes
4.Return Value
On success, a pointer to the memory block allocated by the function.
The type
of this pointer is always void*, which can be cast to the desired type
of data pointer in order to be dereferenceable.
If the function failed to
allocate the requested block of memory, a null pointer is returned.
译:申请内存成功,返回一个指向”被函数分配的内存空间“的指针。
这个指针的类型是void *(指向任意类型的指针),它可以被转换为期望类型的数据指针,以便能够解除引用。
如果函数分配被要求的内存空间失败,则返回一个空指针(NULL)
5.Example
/* malloc example: string generator*/
#include <stdio.h>
#include <stdlib.h>
int main ()
{
int i,n;
char * buffer;
printf ("How long do you want the string? ");
scanf ("%d", &i);
buffer = ();
);
; n<i; n++)
buffer[n]=rand()%+'a';
buffer[i]='\0';
printf ("Random string: %s\n",buffer);
free (buffer);
;
}
6. 注:
1.在#include<stdlib.h>中被定义。
2.malloc是从堆(heap)中分配内存空间,而不是栈(stack)。
3.内存有可能分配失败,所以需要使用下面的检测程序来检测内存是否分配失败。
if(t_array == NULL)
{
printf(Failde to allocate memory block! \n); exit(1); //结束程序
}
4.内存不再使用时,应使用free()函数将内存块释放
5.malloc 函数返回的是 void * 类型,如果你写成:p = malloc (sizeof(int)); 则程序无法通过编译,报错:“不能将 void* 赋值给 int * 类型变量”。
所以必须通过 (int *) 来将强制转换。
6.函数的实参为 sizeof(int) ,用于指明一个整型数据需要的大小。如果你写成:
);
也能通过编译,但事实上只分配了1个字节大小的内存空间,当你往里头存入一个整数,就会有3个字节无家可归,而直接“住进邻居家”!
造成的结果是后面的内存中原有数据内容全部被清空。
7.malloc 只管分配内存,并不能对所得的内存进行初始化,所以得到的一片新内存中,其值将是随机的。
二、使用malloc申请动态内存空间的两种情况:
1. 大容量内存需求
一般需要的内存空间超过0.5M的时候时,要使用malloc来申请内存空间。可以这样理解,因为内存过大,不好管理内存,此时就需要用malloc来管理,而不是IDE。
2.不确定内存需求
当我们需要的内存空间不确定大小的情况下,为了不浪费内存空间,就要用到malloc函数。可以这样理解,如果说静态的内存空间是一个铁盒子的话,那动态内存空间就是一个可收缩的袋子,它的容量可以根据你所装物体的体积不同而改变。而使用静态内存,要么就是申请的内存空间过大,造成浪费,要么就是申请的内存空间不够用,程序异常报错。
三、如何正确使用malloc函数进行动态分配
1.申请一维数组
一维数组的数组名可以看成数组起始元素的首地址,因此我定义一个int *t_array的指针,分配n个大小的int型空间,写法如下:
int * t_array; //定义一维整型指针数组
t_array = (int *)malloc( sizeof(int) * n ); //分配4*n个字节的内存空间
if(t_array == NULL)
{
printf("Failed to allocate memory block! \n");
exit();
}
//具体代码
free(t_array); //内存空间不再用时,释放内存空间 t_array =NULL;
2.申请二维空间
二维数组的数组名是其所有一维数组的首地址,因为二维数组的数组名是指针的指针,因为我定义一个row行column列的二维数组,写法如下:
int ** t_array; //定义二维整型指针数组
int row; //行序号
int column; //列序号
int i;
scanf("%d %d", row,column); //输入行数,列数
t_array = (int **)malloc( sizeof(int *) * row); //分配所有行的首地址 (分配的是行指针单元,每个单元的大小为 sizeof(int *) )
;i<row;i++)
t_array[i] = (int *)malloc(sizeof(int) * column ); //再分配column个整数单元,上面的row个行指针指向这column个整数单元首地址
//具体代码
if(t_array == NULL) //检验是否成功分配内存
{
printf("Failed to allocate memory block! \n");
exit(1);
}
for(i=0;i<row;i++) //检验是否成功分配内存
{
if(t_array[i] == NULL)
{
printf("Failed to allocate memory block! \n");
exit(1);
}
}
for(i=0;i<row;i++) { free(t_array[i]); t_array[i] = NULL; } free(t_array) t_array = NULL;
总结:malloc()函数其实就在内存中找一片指定大小的空间,然后将这个空间的首地址范围给一个指针变量,这里的指针变量可以是一个单独的指针,也可以是一个数组的首地址,这要看malloc()函数中参数size的具体内容。我们这里malloc分配的内存空间在逻辑上连续的,而在物理上可以连续也可以不连续。对于我们程序员来说,我们关注的是逻辑上的连续,因为操作系统会帮我们安排内存分配,所以我们使用起来就可以当做是连续的。
C基础-对malloc的使用与理解的更多相关文章
- IM开发基础知识补课(四):正确理解HTTP短连接中的Cookie、Session和Token
本文引用了简书作者“骑小猪看流星”技术文章“Cookie.Session.Token那点事儿”的部分内容,感谢原作者. 1.前言 众所周之,IM是个典型的快速数据流交换系统,当今主流IM系统(尤其移动 ...
- Objective-C 基础教程第七章,深入理解Xcode
目录 Object-C 基础教程第七章,深入理解Xcode 0x00 前言 0x01 创建工程界面 0x02 主程序界面 ①顶部 Top Test(测试) Profile(动态分析) Analyze( ...
- [C# 基础知识系列]专题九: 深入理解泛型可变性
引言: 在C# 2.0中泛型并不支持可变性的(可变性指的就是协变性和逆变性),我们知道在面向对象的继承中就具有可变性,当方法声明返回类型为Stream,我们可以在实现中返回一个FileStream的类 ...
- 深度学习基础系列(五)| 深入理解交叉熵函数及其在tensorflow和keras中的实现
在统计学中,损失函数是一种衡量损失和错误(这种损失与“错误地”估计有关,如费用或者设备的损失)程度的函数.假设某样本的实际输出为a,而预计的输出为y,则y与a之间存在偏差,深度学习的目的即是通过不断地 ...
- 【java基础】程序员你真的理解反射机制吗?
目录 前言 1.反射的概述 2.正式使用反射之前很有必要了解的Class类 3.反射的使用 前言 很多讲解反射的博客文章并没有详细讲解Class类,~当然包括之前的我也一样~,这样的文章只会让反射徒有 ...
- JavaScript基础知识从浅入深理解(一)
JavaScript的简介 javascript是一门动态弱类型的解释型编程语言,增强页面动态效果,实现页面与用户之间的实时动态的交互. javascript是由三部分组成:ECMAScript.DO ...
- java基础之IO流及递归理解
一.IO流(简单理解是input/output流,数据流内存到磁盘或者从磁盘到内存等) 二.File类(就是操作文件和文件夹的) 1.FIleFile类构造方法 注意:通过构造方法创建的file对象是 ...
- [C# 基础知识系列]专题八: 深入理解泛型(二)
引言: 本专题主要是承接上一个专题要继续介绍泛型的其他内容,这里就不多说了,就直接进入本专题的内容的. 一.类型推断 在我们写泛型代码的时候经常有大量的"<"和"& ...
- Unity 基础-------------------------关于Anchor锚点的理解
Unity进阶技巧 - RectTransform详解 Zui 关注 2016.02.17 01:27 字数 1704 阅读 22157评论 13喜欢 57赞赏 2 RectTransform属性一览 ...
随机推荐
- [luoguP1044] 栈(数论?)
传送门 卡特兰数 代码 #include <cstdio> int n; long long f[20]; int main() { int i; scanf("%d" ...
- 【(待重做)树状数组+dp+离散化】Counting Sequences
https://www.bnuoj.com/v3/contest_show.php?cid=9149#problem/G [题意] 给定一个数组a,问这个数组有多少个子序列,满足子序列中任意两个相邻数 ...
- 利用try-catch代码检查用户输入数据是否是有效的浮点数超级详细
package chapter6; //数据输入格式检查 import java.io.IOException; import java.util.InputMismatchException; im ...
- WCF的Binding模型之四:信道工厂(Channel Factory)
由于信道管理器在客户端和服务端所起的不同作用,分为信道监听器和信道工厂.和服务端的信道监听其相比,处于客户端的信道工厂显得简单.从名称就可以看得出来,信道工厂的作用就是单纯的创建用于消息发送的信道.我 ...
- MongoDB小结27 - 聚合管道【$project】
我们有这样的数据 { "_id" : 1, title: "abcdef", isbn: "6969696969", author: { l ...
- 根据身份证号,取得行政区划的Javascript实现
原文:http://www.cnblogs.com/baibaluo/archive/2011/06/03/2071255.html#2585076 项目里需要一个根据身份证号,取得发证地行政区划的的 ...
- component and slot
component and slot 使用: 1.component panel <article class="message"> <div class=&qu ...
- 怎样载入指定路径的Logback.xml
今天想外置logback.xml,谢了例如以下代码 File logbackFile = new File("./conf/logback.xml"); if (logbackFi ...
- CentOS 5.5 介绍
转载:http://www.osyunwei.com/archives/3625.html 操作系统:CentOS 5.5主机名称:wwwIP地址:192.168.21.128网关:192.168.2 ...
- Androidbuttonshape形状资源码实现
1.项目Src下创建drawable 看文档Develop/API Guides/App Resources/Drawable/Shape Drawable 单词:corners : 角 ; gr ...