一、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. 常见问题:Linux安装Python3步骤、Windows无法利用pip

    Linux安装python3.6和第三方库的步骤: 我的Linux是CentOS 6.5版本 Linux下大部分系统默认自带python2.x的版本,最常见的是python2.6或python2.7, ...

  2. EasyUI单击行数据时动态编写editor

    $.extend($.fn.treegrid.methods, { addEditor: function (jq, param) { if (param instanceof Array) { $. ...

  3. POJ3728 The merchant解题报告

    Description There are N cities in a country, and there is one and only one simple path between each ...

  4. symfony 使用原始sql

    $this->get('database_connection')->fetchAll('select * from book where book.id=3')

  5. Python学习系列之内置函数

    数学相关 abs(a):求取绝对值 max(list):求取list最大值 min(list):求取list最小值 sum(list):求取list元素的和 sorted(list):排序,返回排序后 ...

  6. 006 Cisco switch prewired

    Switch>en Switch#config t Enter configuration commands, one per line.  End with CNTL/Z. Switch(co ...

  7. 查找olr备份路径

    使用:ocrdump -local <olr_dump_name> more <olr_dump_name> 来查找 [SYSTEM.OLR.BACKUP.LOC] 的相应键值 ...

  8. 浏览器判断是否安装了ios/android客户端程序

    最近在做一个项目,该项目的前身是为mobile browser量身打造的一个网站.现在有这样一个需求: 当用户在用mobile browser浏览该网站的时候会点击一个按钮/超链接,通过这个按钮的点击 ...

  9. .NET中的PublicKeyToken以及强命名问题

    在.NET的GAC出现之前,以前有DLL Hell的问题.这是由于当时对于共享的DLL的处理方式.是通过採用注冊表的方式实现的.当我们安装一个程序A的时候,这个程序包括一个共享的DLL,那么这个DLL ...

  10. FreeRTOS系列第14篇---FreeRTOS任务通知

    注:本文介绍任务通知的基础知识,具体源代码分析见<FreeRTOS高级篇8---FreeRTOS任务通知分析> 每一个RTOS任务都有一个32位的通知值,任务创建时,这个值被初始化为0.R ...