1.   本文介绍mallocfree函数的内容。
  2.  
  3.   C中,对内存的管理是相当重要。下面开始介绍这两个函数:
  4.  
  5.   一、malloc()和free()的基本概念以及基本用法:
  6.  
  7. 、函数原型及说明:
  8.  
  9. void *malloc(long NumBytes):该函数分配了NumBytes个字节,并返回了指向这块内存的指针。如果分配失败,则返回一个空指针(NULL)。
  10.  
  11. 关于分配失败的原因,应该有多种,比如说空间不足就是一种。
  12.  
  13. void free(void *FirstByte): 该函数是将之前用malloc分配的空间还给程序或者是操作系统,也就是释放了这块内存,让它重新得到自由。
  14.  
  15. 、函数的用法:
  16.  
  17. 其实这两个函数用起来倒不是很难,也就是malloc()之后觉得用够了就甩了它把它给free()了,举个简单例子:
  18.  
  19. 程序代码:
  20. // Code...
  21. char *Ptr = NULL;
  22. Ptr = (char *)malloc( * sizeof(char));
  23. if (NULL == Ptr)
  24.   {
  25.    exit ();
  26.    }
  27. gets(Ptr);
  28.  
  29. // code...
  30. free(Ptr);
  31. Ptr = NULL;
  32. // code...
  33.  
  34. 就是这样!当然,具体情况要具体分析以及具体解决。比如说,你定义了一个指针,在一个函数里申请了一块内存然后通过函数返回传递给这个指针,那么也许释放这块内存这项工作就应该留给其他函数了。
  35.  
  36. 、关于函数使用需要注意的一些地方:
  37.  
  38. A、申请了内存空间后,必须检查是否分配成功。
  39.  
  40. B、当不需要再使用申请的内存时,记得释放;释放后应该把指向这块内存的指针指向NULL,防止程序后面不小心使用了它。
  41.  
  42. C、这两个函数应该是配对。如果申请后不释放就是内存泄露;如果无故释放那就是什么也没有做。释放只能一次,如果释放两次及两次以上会
  43.  
  44. 出现错误(释放空指针例外,释放空指针其实也等于啥也没做,所以释放空指针释放多少次都没有问题)。
  45.  
  46. D、虽然malloc()函数的类型是(void *),任何类型的指针都可以转换成(void *),但是最好还是在前面进行强制类型转换,因为这样可以躲过一些编译器的检查。
  47.  
  48. 现在进入第二部分:
  49.  
  50.   二、malloc()到底从哪里得来了内存空间:
  51.  
  52. malloc()到底从哪里得到了内存空间?答案是从堆里面获得空间。也就是说函数返回的指针是指向堆里面的一块内存。操作系统中有一个记录空闲内存地址的链表。当操作系统收到程序的申请时,就会遍历该链表,然后就寻找第一个空间大于所申请空间的堆结点,然后就将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。关于堆的知识呢可以查询数据结构方面的知识或查询以前的一篇帖子C/C++堆、栈及静态数据区详解[转载]。这里不过多介绍。
  53.  
  54. 、在使用malloc()分配内存空间后,一定要记得释放内存空间,否则就会出现内存泄漏。
  55.  
  56. free()到底释放了什么
  57.  
  58. free()释放的是指针指向的内存!注意!释放的是内存,不是指针!指针并没有被释放,指针仍然指向原来的存储空间。指针是一个变量,只有程序结束时才被销毁。释放了内存空间后,原来指向这块空间的指针还是存在!只不过现在指针指向的内容的垃圾,是未定义的,所以说是垃圾。因此,释放内存后把指针指向NULL,防止指针在后面不小心又被解引用了。
  59.  
  60.   三、malloc()以及free()的机制:
  61.  
  62. 事实上,仔细看一下free()的函数原型,也许也会发现似乎很神奇,free()函数非常简单,只有一个参数,只要把指向申请空间的指针传递给free()中的参数就可以完成释放工作!这里要追踪到malloc()的申请问题了。申请的时候实际上占用的内存要比申请的大。因为超出的空间是用来记录对这块内存的管理信息。
  63.  
  64. 大多数实现所分配的存储空间比所要求的要稍大一些,额外的空间用来记录管理信息——分配块的长度,指向下一个分配块的指针等等。这就意味着如果写过一个已分配区的尾端,则会改写后一块的管理信息。这种类型的错误是灾难性的,但是因为这种错误不会很快就暴露出来,所以也就很难发现。将指向分配块的指针向后移动也可能会改写本块的管理信息。
  65.  
  66. malloc()申请的空间实际就是分了两个不同性质的空间。一个就是用来记录管理信息的空间,另外一个就是可用空间了。而用来记录管理信息的实际上是一个结构体。在C语言中,经常用结构来记录信息!下面看看这个结构体的原型:
  67.  
  68. 程序代码:
  69. struct mem_control_block {
  70. int is_available; //一般来说应该是一个可用空间的首地址,但这里英文单词却显示出空间是否可用的一个标记
  71. int size; //这是实际空间的大小
  72. };
  73.  
  74.   所以,free()就是根据这个结构体的信息来释放malloc()申请的空间!而结构体的两个成员的大小我想应该是操作系统的事了。
  75.   下面看看free()的源代码
  76.    // code...
  77.  
  78. void free(void *ptr)
  79. {
  80. struct mem_control_block *free;
  81. free = ptr - sizeof(struct mem_control_block);
  82. free->is_available = ;
  83. return;
  84. }
  85.  
  86.   至于malloc的源码,有兴趣的可以到网上找一下!

free与malloc详解

以上摘自:https://www.cnblogs.com/hanyonglu/archive/2011/04/28/2031271.html

在做测试时遇到一个小问题,增加了对free与malloc的一些认识。

1.  L = create_seqlist  //创建一个线性表

      L = (seqlist *)malloc(sizeof(seqlist))

2.  insert_seqlist  //循环插入元素

3.  show_seqlist(L)  //打印线性表的内容

4.  clear_seqlist(L)  //q清除线性表

       free(L)

5.   show_seqlist(L) //打印线性表的内容

运行后发现clear后,依然能打印出线性表的内容

为什么free了之后,L指针指向的内容还在,么有改变?

首先malloc向系统申请了内存,并把内存地址赋给了一个指针L,之后通过L就可以对内存进行操作,并且这部分内存就不能通过其他指针被访问。
free是指释放L指针指向的内存,即把malloc申请的空间的使用权限交还给系统,并非释放指针L,就是说L指针的地址还是指向之前分配的那块内存,但是此内存也可以被其他指针访问,修改。指针L是局部变量,在所在函数调用结束后会被释放,使用权交还系统,但L指针空间的数据不会改变,除非有人修改,所以在show函数再次调用,还能打印出顺序表的内容。为了防止非法访问,一般在free后会将指针置NULL。

对于上程序,在clear中加 L =  NULL,置空指针并没有用,因为L指针也属于main函数中的局部变量,在clear调用结束后,L指向内容不变。

对malloc与free函数的浅识的更多相关文章

  1. malloc 与 free函数详解<转载>

    malloc和free函数详解   本文介绍malloc和free函数的内容. 在C中,对内存的管理是相当重要.下面开始介绍这两个函数: 一.malloc()和free()的基本概念以及基本用法: 1 ...

  2. ZH奶酪:C语言中malloc()和free()函数解析

    1.malloc()和free()的基本介绍 (1)函数原型及说明 void *malloc(long NumBytes) 该函数分配了NumBytes个字节,并返回了指向这块内存的指针.如果分配失败 ...

  3. malloc与free函数用法

    在C里,内存管理是通过专门的函数来实现.另外,为了兼容各种编程语言,操作系统提供的接口通常是 C 语言写成的函数声明 (Windows 本身也由C和汇编语言写成). 1 分配内存 malloc 函数 ...

  4. 关于malloc和free函数的用法

    原文:http://blog.pfan.cn/vfdff/33507.html 个人总结 在C语言的学习中,对内存管理这部分的知识掌 握尤其重要!之前对C中的malloc()和free()两个函数的了 ...

  5. 对c语言中malloc和free函数的理解

    最近在复习c语言的时候再次用到了malloc函数和free函数,此处着讲解一下自己对这两个函数的理解和认识. 一. malloc函数和free函数的基本概念和基本的用法 对于malloc函数: 1.  ...

  6. malloc和free函数详解

    一.malloc()和free()的基本概念以及基本用法: 1.函数原型及说明: void *malloc(long NumBytes):该函数分配了NumBytes个字节,并返回了指向这块内存的指针 ...

  7. 使用malloc和free函数进行内存动态分配

    一.在学习c语言里面,内存分配这个话题非常有意思,因为我们平时在开发的时候,如果一不小心没注意内存释放的话,写的的程序很容易出错,所以今天就来回顾一下c语言里面的内存动态分配,下面我们先来看一个实例来 ...

  8. 关于 block的一些浅识

    block的定义:“带自动变量的匿名函数” (一)写法: ^ void (int iAge){ NSLog(@"%d", iAge);}; 和C函数写法区别在于: 1) :以插入符 ...

  9. malloc和free函数的使用

    #include <stdio.h> #include <stdlib.h> int main() { int *p,t; p = (int *)malloc(40*sizeo ...

随机推荐

  1. 【串线篇】REST风格的请求格式

    1.什么是rest 答出这两点就够了: 1.1 统一接口 rest其实是基于HTTP的,四种方式. RESTful架构风格规定,数据的元操作,即CRUD(create, read, update和de ...

  2. Mybatis基于XML配置SQL映射器(一)

    Durid和Mybatis开发环境搭建 SpringBoot搭建基于Spring+SpringMvc+Mybatis的REST服务(http://www.cnblogs.com/nbfujx/p/76 ...

  3. UNP学习第六章select

    一.I/O复用典型的网络应用场合 当客户处理多个描述字时,必须使用I/O复用,这在前一段中已做了描述. 一个客户同时处理多个套接口时可能的,但很少出现. 如果一个TCP服务器既要处理监听套接口,又要处 ...

  4. docker-compose快速搭建 Prometheus+Grafana监控系统

    一.说明Prometheus负责收集数据,Grafana负责展示数据.其中采用Prometheus 中的 Exporter含:1)Node Exporter,负责收集 host 硬件和操作系统数据.它 ...

  5. (转)Maven创建webapp项目无法修改web版本的问题

    maven创建的web app,默认使用的servlet版本是2.3,默认不支持JSTL,为了默认支持JSTL表达式,需要升级servlet到3.0 转:http://blog.sina.com.cn ...

  6. ASP.NET Error Handling

    https://docs.microsoft.com/en-us/aspnet/web-forms/overview/getting-started/getting-started-with-aspn ...

  7. java中 抽象类和接口的区别

    一. 什么是抽象类及什么是抽象方法 抽象方法是一种特殊的方法:他只有声明,而没有具体实现,抽象方法的声明格式为: abstract void funName(); 抽象方法必须用 abstract 修 ...

  8. HTML-参考手册: 画布

    ylbtech-HTML-参考手册: 画布 1.返回顶部 1. HTML5 <canvas> 参考手册 描述 HTML5 <canvas> 标签用于绘制图像(通过脚本,通常是 ...

  9. 54、salesforce学习笔记(一)

    Decimal priceDecimal = -4.50; System.debug('小数的绝对值为:'+priceDecimal.abs()); System.debug('priceDecima ...

  10. AutoMapper用法 转载https://www.cnblogs.com/youring2/p/automapper.html

    AutoMapper是对象到对象的映射工具.在完成映射规则之后,AutoMapper可以将源对象转换为目标对象. 配置AutoMapper映射规则 AutoMapper是基于约定的,因此在实用映射之前 ...