[X]关闭

C语言如何向系统接要存

文档创建者:宋桓公
浏览次数:4551
最后更新:2015-11-27
C语言如何向系统接要存,就有这么三种方式:
1、向栈要。     
2、向堆要。     
3、向数据段要。     
这一下就扯出了三种内存空间,内存空间的本质是一样的,一个地址对应一个方框,方框里可以放数据。但是为了更好的去     
管理这篇空间,于是将空间分成了几个不同的区域,这些区域有:     
1、栈:存放局部变量。     
2、堆:存放malloc分配的变量。     
3、数据段,如果数据段继续进行细分,还可以分为:     
        3.1 .data 段 放置初值不为0的全局变量。     
        3.2 .bss  段 放置初值为0,或者没赋初值的全局变量。(他们的结果是一样的,初值为0,应为.bss会被编译器清理)     
4、代码段:代码段是存放(能够改变的变量之外的东西)代码的,虽然也是在内存中,但是是只读的,程序员无法去修改它。     
所以在讲向系统要内存时,我并没有说到这个内存区域。     
//--栈     
那我们就一个个的说,先说栈,这片区域是有栈管理器来管理的(栈管理器其实就是一种算法,一段代码)。当我们     
申请局部变量的时候,其实就是在向栈要内存。使用栈的好处是,自动化管理,使用方便。由于分配的空间是连续的,     
所以管理起来也比较方便,用起来效率也比较高。(相对于堆)。     
//--堆     
再看堆,堆也是一片内存空间,由堆管理器掌管。C语言中用malloc分配的变量空间,就会到堆上。但分配的空间不一定是     
连续的,这是为了解决栈的一个缺点——应为栈分配空间是连续的,而且有需要被反复使用,这样会造成空间碎片无法利用。     
而堆管理器使用链表的方式解决了这个问题,可以充分的利益空间,缺点当然就是使用起来没那么方便,而且效率也没有栈高。     
但是,从堆和栈的空间大小来看,此时效率的这点差异,远没有空间利用率的价值大。堆这块内存区域往往比栈这块内存区域大很多。     
这也是为什么,在申请一个很大的内存空间时,需要在堆上,而不是栈上。
//--数据段   
数据段是存放全局变量的。细分可以分为.data和 .bss 这个已经说过了。     
数据段,是由谁管理的? 堆有堆管理器,栈有栈管理器。数据段呢? 这个问题,我不清楚~~     
有一个说法是:数据段可读可写不可执行,这个我也似懂非懂~~
     
反正数据段是放全局变量的,多的我也不懂,就说这么多。
//--代码段   
最后看代码段,代码段的特点是放一些我们改不了的东西,如程序代码等。所以上图中,说他可读,可执行,但是不可写。     
代码段朱老师讲的很清楚了,我就直接摘录笔记了:     
/*******************************************************************************     
代码段:代码段就是程序中的可执行部分,直观理解代码段就是函数堆叠组成的。     
4.5.4.2、有些特殊数据会被放到代码段     
(1)C语言中使用char *p = "linux";定义字符串时,字符串"linux"实际被分配在代码段,     
也就是说这个"linux"字符串实际上是一个常量字符串而不是变量字符串。     
(2)const型常量:C语言中const关键字用来定义常量,常量就是不能被改变的量。     
const的实现方法至少有2种:第一种就是编译将const修饰的变量放在代码段去以实现不能修改(普遍见于各种单片机的编译器);     
第二种就是由编译器来检查以确保const型的常量不会被修改,实际上const型的常量还是和普通变量一样放在数据段的(gcc中就是这样实现的)。     
/*******************************************************************************     
这里就说下我的理解:     
1、有些时候常量会被直接放到代码段,这样程序员就无法修改了。如字符串它就是个常量,他就是被放到代码段,     
你程序员是无法修改它的。修改了运行时就会报段错误!(注意这里编译不会报错,如果写成const char *p = "linux",编译时就会报错     
所以明显这么写更好!)     
2、有的变量本身不是常量,但是加上const修饰后就变成常量了。如果编译器吧const修饰的变量都丢到代码段,程序员当然     
就无法修改,但是GCC不是这么做的,他并没有将const修饰的变量放到代码段。所以const修饰的变量不是不能被修改,只是不能     
直接被修改。如用指针可以间接修改。不过编译器会警告。GCC的做法就是——警告你不要跳楼,但是你非要调,我也不拦你。。。
那么,这四个区域就讲完了。现在就讨论下数据段中的全局变量和堆中的变量有什么区别:   
1、生命周期不同     
    “堆内存的生命周期是从malloc开始到free结束,而全局变量是从整个程序一开始执行就开始,     
直到整个程序结束才会消灭,伴随程序运行的一生。”——引用朱友鹏笔记     
2、用法不同。     
    数据段上的全局变量,是在程序编译的时候就确定了,程序运行时大小是不可更改的。所以数据段也通常被称为静态区域。     
而堆上的变量是程序运行才分配的(C语言通过函数malloc分配),也觉是说它是动态的,说白了就是malloc里可以放一个变量,     
达到根据程序运行的情况,起到一个随机应变的效果。     
3、大小不同     
我想堆的区域应该比数据段大很多。
总结:   
1、分析了半天也就是堆的好处最多,但是还没说他的使用方法,那就简单演示下:     
char *p = (char *)malloc(10);   
    if (NULL == p)     
    {     
        printf("malloc error.\n");     
        return -1;     
    }     
    memset(p, 0, 10);        // 第三种方法: 放在malloc申请的堆内存中     
    strcpy(p, "linux");     
2、至于数据段的管理,确实不是很清楚,只知道通过全局指针访问(全局指针,ARM中就是PC指针)。(内心语:全局指针访问全局变量     
这也是挺匹配的哈!栈有栈指针,堆有链表,有意思。)     
3、现在不明白的东西,也不要太纠结,现在实在弄不懂就先放着,说明自己还没到那个程度     
不要忘记自己的学习主线才是最重要的,所以说不求甚解也是一种智慧~~!!

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则