浅谈c语言中的堆

操作系统堆管理器管理:

堆管理器是操作系统的一个模块,堆管理内存分配灵活,按需分配。
 大块内存:

  堆内存管理者总量很大的操作系统内存块,各进程可以按需申请使用,使用完释放。
 程序手动申请&释放:

  手工意思是需要写代码去申请malloc和释放free。
 脏内存:

  堆内存也是反复使用的,而且使用者用完释放前不会清除,因此也是脏的。
 临时性:

  堆内存只在malloc和free之间属于我这个进程,而可以访问。在malloc之前和free之后
      都不能再访问,否则会有不可预料的后果。

堆内存使用范例

(1)void *是个指针类型,malloc返回的是一个void *类型的指针,实质上malloc返回的是堆管理器分配给我本次申请的那段内存空间的首地址(malloc返回的值其实是一个数字,这个数字表示一个内存地址)。为什么要使用void *作为类型?主要原因是malloc帮我们分配内存时只是分配了内存空间,至于这段空间将来用来存储什么类型的元素malloc是不关心的,由我们程序自己来决定。

(2)什么是void类型。void类型不表示没有类型,而表示万能类型。void的意思就是说这个数据的类型当前是不确定的,在需要的时候可以再去指定它的具体类型。void *类型是一个指针类型,这个指针本身占4个字节,但是指针指向的类型是不确定的,换句话说这个指针在需要的时候可以被强制转化成其他任何一种确定类型的指针,也就是说这个指针可以指向任何类型的元素。

(3)malloc的返回值:成功申请空间后返回这个内存空间的指针,申请失败时返回NULL。所以malloc获取的内存指针使用前一定要先检验是否为NULL。

(4)malloc申请的内存时用完后要free释放。free(p);会告诉堆管理器这段内存我用完了你可以回收了。堆管理器回收了这段内存后这段内存当前进程就不应该再使用了。因为释放后堆管理器就可能把这段内存再次分配给别的进程,所以你就不能再使用了。

(5)再调用free归还这段内存之前,指向这段内存的指针p一定不能丢(也就是不能给p另外赋值)。因为p一旦丢失这段malloc来的内存就永远的丢失了(内存泄漏),直到当前程序结束时操作系统才会回收这段内存。

(6) malloc(0) malloc申请0字节内存本身就是一件无厘头事情,一般不会碰到这个需要。 如果真的malloc(0)返回的是NULL还是一个有效指针?答案是:实际分配了16Byte的一段内存并且返回了这段内存的地址。这个答案不是确定的,因为C语言并没有明确规定malloc(0)时的表现,由各malloc函数库的实现者来定义。

  malloc(4) gcc中的malloc默认最小是以16B为分配单位的。如果malloc小于16B的大小时都会返回一个16字节的大小的内存。malloc实现时没有实现任意自己的分配而是允许一些大小的块内存的分配。    malloc(20)去访问第25、第250、第2500····会怎么样 实战中:120字节处正确,1200字节处正确····终于继续往后访问总有一个数字处开始段错误了。

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3
 4
 5 int main(void)
 6 {
 7     int *p = (int *)malloc(20);
 8     // 第二步:检验分配是否成功
 9     if (NULL == p)
10     {
11         printf("malloc error.\n");
12         return -1;
13     }
14
15     *(p+3) = 12;
16     *(p+300000) = 1234;
17     printf("*(p+3) = %d.\n", *(p+3));
18     printf("*(p+300000) = %d.\n", *(p+300000));
19
20
21
22 /*
23     int *p1 = (int *)malloc(4);        // p2-p1 = 0x10 = 16Byte
24     int *p2 = (int *)malloc(4);
25
26     printf("p1 = %p.\n", p1);        // p2-p1 = 0x10 = 16Byte
27     printf("p2 = %p.\n", p2);
28 */
29
30 /*
31     int *p1 = (int *)malloc(0);
32     int *p2 = (int *)malloc(0);
33
34     printf("p1 = %p.\n", p1);        // p2-p1 = 0x10 = 16Byte
35     printf("p2 = %p.\n", p2);
36     */
37 /*
38     // 需要一个1000个int类型元素的数组
39
40     // 第一步:申请和绑定
41     int *p = (int *)malloc(1000*sizeof(int));
42     // 第二步:检验分配是否成功
43     if (NULL == p)
44     {
45         printf("malloc error.\n");
46         return -1;
47     }
48
49     // 第三步:使用申请到的内存
50     //p = NULL;
51     //p = &a;    // 如果在free之前给p另外赋值,那么malloc申请的那段内存就丢失掉了
52                 // malloc后p和返回的内存相绑定,p是那段内存在当前进程的唯一联系人
53                 // 如果p没有free之前就丢了,那么这段内存就永远丢了。丢了的概念就是
54                 // 在操作系统的堆管理器中这段内存是当前进程拿着的,但是你也用不了
55                 // 所以你想申请新的内存来替换使用,这就叫程序“吃内存”,学名叫内存泄漏
56     *(p+0) = 1;
57     *(p+1) = 2;
58     printf("*(p+0) = %d.\n", *(p+0));
59     printf("*(p+1) = %d.\n", *(p+1));
60
61     *(p+222) = 133;
62     *(p+223) = 222;
63
64
65     // 第四步:释放
66     free(p);
67     p = NULL;
68
69     printf("*(p+222) = %d.\n", *(p+222));
70     printf("*(p+223) = %d.\n", *(p+223));
71 */
72     return 0;
73 }
时间: 2024-10-23 21:40:03

浅谈c语言中的堆的相关文章

浅谈C语言中的联合体(转载)

联合体union 当多个数据需要共享内存或者多个数据每次只取其一时,可以利用联合体(union).在C Programming Language 一书中对于联合体是这么描述的: 1)联合体是一个结构: 2)它的所有成员相对于基地址的偏移量都为0: 3)此结构空间要大到足够容纳最"宽"的成员: 4)其对齐方式要适合其中所有的成员: 下面解释这四条描述: 由于联合体中的所有成员是共享一段内存的,因此每个成员的存放首地址相对于于联合体变量的基地址的偏移量为0,即所有成员的首地址都是一样的.为

浅谈C语言中的强符号、弱符号、强引用和弱引用

摘自http://www.jb51.net/article/56924.htm 浅谈C语言中的强符号.弱符号.强引用和弱引用 投稿:hebedich 字体:[增加 减小] 类型:转载 时间:2014-10-31 我要评论 这篇文章主要介绍了C语言中的强符号.弱符号.强引用和弱引用的定义及相关内容,非常的简单易懂,有需要的朋友可以参考下 首先我表示很悲剧,在看<程序员的自我修养--链接.装载与库>之前我竟不知道C有强符号.弱符号.强引用和弱引用.在看到3.5.5节弱符号和强符号时,我感觉有些困惑

浅谈 C 语言中模块化设计的范式

今天继续谈模块化的问题.这个想慢慢写成个系列,但是不一定连续写.基本是想起来了,就整理点思路出来.主要还是为以后集中整理做点铺垫. 我们都知道,层次分明的代码最容易维护.你可以轻易的换掉某个层次上的某个模块,而不用担心对整个系统造成很大的副作用. 层次不清的设计中,最糟糕的一种是模块循环依赖.即,分不清两个模块谁在上,谁在下.这个时候,最容易牵扯不清,其结果往往是把两者看做一体去维护算了.这里面还涉及一些初始化次序等繁杂的细节. 其次,就是越层的模块联系.当模块 A 是模块 B 的上层,而模块

浅谈C语言中的联合体

联合体union 当多个数据需要共享内存或者多个数据每次只取其一时,可以利用联合体(union).在C Programming Language 一书中对于联合体是这么描述的: 1)联合体是一个结构: 2)它的所有成员相对于基地址的偏移量都为0: 3)此结构空间要大到足够容纳最"宽"的成员: 4)其对齐方式要适合其中所有的成员: 下面解释这四条描述: 由于联合体中的所有成员是共享一段内存的,因此每个成员的存放首地址相对于于联合体变量的基地址的偏移量为0,即所有成员的首地址都是一样的.为

浅谈C#语言中的各种数据类型,与数据类型之间的转换

什么是数据类型? 数据类型,百度百科是这样解释的:数据类型在数据结构中的定义是一个值的集合以及定义在这个值集上的一组操作.这样的解释对于一个初学者来说未必太过于深奥. 简单点说,数据类型就是不同长度的数据的归类.数据类型的出现就是为了解决计算机中对不同长度的数据合理保存的问题.为了把数据分成所需内存大小不同的数据,编程的时候需要用大数据的时候才需要申请大内存,就可以充分利用内存.达到资源合理优化利用,减少浪费内存的目的. 数据类型有哪些? 类型            描述            

浅谈C语言中结构体的初始化

转自:http://www.jb51.net/article/37246.htm <代码大全>建议在变量定义的时候进行初始化,但是很多人,特别是新人对结构体或者结构体数组定义是一般不会初始化,或者不知道怎么初始化.1.初始化 复制代码代码如下: typedef struct _TEST_T {        int i;        char c[10];}TEST_T;TEST_T gst  = {1, “12345”};//可以初始化,设置i为1,s为一个字符串.TEST_T gst 

转: 浅谈C/C++中的指针和数组(二)

转自:http://www.cnblogs.com/dolphin0520/archive/2011/11/09/2242419.html 浅谈C/C++中的指针和数组(二) 前面已经讨论了指针和数组的一些区别,然而在某些情况下,指针和数组是等同的,下面讨论一下什么时候指针和数组是相同的. C语言标准对此作了说明: 规则1:表达式中的数组名被编译器当做一个指向该数组第一个元素的指针: 注:下面几种情况例外 1)数组名作为sizeof的操作数 2)使用&取数组的地址 规则2:下标总是与指针的偏移量

浅谈SQL Server中的事务日志(一)----事务日志的物理和逻辑构架

浅谈SQL Server中的事务日志(一)----事务日志的物理和逻辑构架 简介 SQL Server中的事务日志无疑是SQL Server中最重要的部分之一.因为SQL SERVER利用事务日志来确保持久性(Durability)和事务回滚(Rollback).从而还部分确保了事务的ACID属性.在SQL Server崩溃时,DBA还可以通过事务日志将数据恢复到指定的时间点.当SQL Server运转良好时,多了解一些事务日志的原理和概念显得并不是那么重要.但是,一旦SQL SERVER发生崩

浅谈前、中、后缀表达式

浅谈前.中.后缀表达式 前.中.后缀表达式是信息学奥林匹克竞赛中比较鸡肋的知识点.但是知识点在考纲范围内,而且中缀表达式转后缀表达式是比较有用的知识.所以在这里为大家简单介绍一下. 之前在自学前.中.后缀表达式的时候,发现网上的很多博客和讲解的思路都不是很明了,或者就是对新手不是很友好,感谢@JZYShurak的讲解,让我对这个东西建立了一个直观的认识.所以我来补一篇比较基础,比较好理解,语言比较简洁的博客.希望能对各路大佬有所些许的帮助. 中缀表达式 中缀表达式就是我们生活中常用的表达式,简单