一、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的使用与理解的更多相关文章

  1. IM开发基础知识补课(四):正确理解HTTP短连接中的Cookie、Session和Token

    本文引用了简书作者“骑小猪看流星”技术文章“Cookie.Session.Token那点事儿”的部分内容,感谢原作者. 1.前言 众所周之,IM是个典型的快速数据流交换系统,当今主流IM系统(尤其移动 ...

  2. Objective-C 基础教程第七章,深入理解Xcode

    目录 Object-C 基础教程第七章,深入理解Xcode 0x00 前言 0x01 创建工程界面 0x02 主程序界面 ①顶部 Top Test(测试) Profile(动态分析) Analyze( ...

  3. [C# 基础知识系列]专题九: 深入理解泛型可变性

    引言: 在C# 2.0中泛型并不支持可变性的(可变性指的就是协变性和逆变性),我们知道在面向对象的继承中就具有可变性,当方法声明返回类型为Stream,我们可以在实现中返回一个FileStream的类 ...

  4. 深度学习基础系列(五)| 深入理解交叉熵函数及其在tensorflow和keras中的实现

    在统计学中,损失函数是一种衡量损失和错误(这种损失与“错误地”估计有关,如费用或者设备的损失)程度的函数.假设某样本的实际输出为a,而预计的输出为y,则y与a之间存在偏差,深度学习的目的即是通过不断地 ...

  5. 【java基础】程序员你真的理解反射机制吗?

    目录 前言 1.反射的概述 2.正式使用反射之前很有必要了解的Class类 3.反射的使用 前言 很多讲解反射的博客文章并没有详细讲解Class类,~当然包括之前的我也一样~,这样的文章只会让反射徒有 ...

  6. JavaScript基础知识从浅入深理解(一)

    JavaScript的简介 javascript是一门动态弱类型的解释型编程语言,增强页面动态效果,实现页面与用户之间的实时动态的交互. javascript是由三部分组成:ECMAScript.DO ...

  7. java基础之IO流及递归理解

    一.IO流(简单理解是input/output流,数据流内存到磁盘或者从磁盘到内存等) 二.File类(就是操作文件和文件夹的) 1.FIleFile类构造方法 注意:通过构造方法创建的file对象是 ...

  8. [C# 基础知识系列]专题八: 深入理解泛型(二)

    引言: 本专题主要是承接上一个专题要继续介绍泛型的其他内容,这里就不多说了,就直接进入本专题的内容的. 一.类型推断 在我们写泛型代码的时候经常有大量的"<"和"& ...

  9. Unity 基础-------------------------关于Anchor锚点的理解

    Unity进阶技巧 - RectTransform详解 Zui 关注 2016.02.17 01:27 字数 1704 阅读 22157评论 13喜欢 57赞赏 2 RectTransform属性一览 ...

随机推荐

  1. [luoguP1044] 栈(数论?)

    传送门 卡特兰数 代码 #include <cstdio> int n; long long f[20]; int main() { int i; scanf("%d" ...

  2. 【(待重做)树状数组+dp+离散化】Counting Sequences

    https://www.bnuoj.com/v3/contest_show.php?cid=9149#problem/G [题意] 给定一个数组a,问这个数组有多少个子序列,满足子序列中任意两个相邻数 ...

  3. 利用try-catch代码检查用户输入数据是否是有效的浮点数超级详细

    package chapter6; //数据输入格式检查 import java.io.IOException; import java.util.InputMismatchException; im ...

  4. WCF的Binding模型之四:信道工厂(Channel Factory)

    由于信道管理器在客户端和服务端所起的不同作用,分为信道监听器和信道工厂.和服务端的信道监听其相比,处于客户端的信道工厂显得简单.从名称就可以看得出来,信道工厂的作用就是单纯的创建用于消息发送的信道.我 ...

  5. MongoDB小结27 - 聚合管道【$project】

    我们有这样的数据 { "_id" : 1, title: "abcdef", isbn: "6969696969", author: { l ...

  6. 根据身份证号,取得行政区划的Javascript实现

    原文:http://www.cnblogs.com/baibaluo/archive/2011/06/03/2071255.html#2585076 项目里需要一个根据身份证号,取得发证地行政区划的的 ...

  7. component and slot

    component and slot 使用: 1.component panel <article class="message"> <div class=&qu ...

  8. 怎样载入指定路径的Logback.xml

    今天想外置logback.xml,谢了例如以下代码 File logbackFile = new File("./conf/logback.xml"); if (logbackFi ...

  9. CentOS 5.5 介绍

    转载:http://www.osyunwei.com/archives/3625.html 操作系统:CentOS 5.5主机名称:wwwIP地址:192.168.21.128网关:192.168.2 ...

  10. Androidbuttonshape形状资源码实现

    1.项目Src下创建drawable 看文档Develop/API Guides/App Resources/Drawable/Shape Drawable 单词:corners : 角  ;  gr ...