C语言程序内存四区(栈区,堆区,全局区,代码区)

一,综述 
在C语言程序中, 代码在内存中进行执行的时候,我们粗略的将程序所占用的内存分为四个区域—-栈区,堆区,全局区,代码区. 
每个程序都有唯一的四个内存区域.我们需要熟悉和了解各个区域的特性,例如存储什么类型的数据, 有谁去申请开辟, 又有谁去管理释放等等

二,内存四区介绍

<一>栈区 
由编译器自动分配释放, 存放函数的参数值,局部变量等. 
例如: 参数buf,参数bufsize和size都是存放在栈区.当函数执行完毕的时候,自动释放

void  recev(char* buf, int bufsize){
            int size;
}
 
 
  • 1
  • 2
  • 3

<二> 堆区 
一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回收

例如:下面的src所指向的内存空间就是在堆区

char* src = (char*) malloc(sizeof(buf) * sizeof(10));
 
 
  • 1

<三>全局区(静态区) 
全局变量和静态变量存放在此. 里面细分有一个常量区, 字符串常量和其他常量也存放在此. 该区域是在程序结束后由操作系统释放.

<四> 程序代码区 
这个区域存放函数体的二进制代码.也是由操作系统进行管理的

三, 划分内存四区的意义 
C语言程序中,根据是局部变量,全局变量, 常量还是通过malloc等类似的函数分配内存空间, 把他们放到对应的内存区中.这样就赋予了这些变量或常量不同的生命周期, 不同的释放方式. 根据我们程序的需要,我们在编码过程中,声明不同的变量类型, 使他们有不同的声明长度, 不同的释放方式,给我们更大的灵活编程

四,内存四区的一个示例

代码如下:

/*
变量生命周期示例
*/
#include "stdafx.h"
#include <stdio.h>
#include<stdlib.h>

/*
访问栈区
*/
int visitStackArea(){
    printf("-----------------visitStackArea()---------------------\n");
    int a = 10;
    printf("a的地址 %d\n",&a);
    return a;
}

/*
访问全局区
*/
char* visitGlobalArea(){
    printf("----------------visitGlobalArea()----------------------\n");
    char *str = "abcdefg";
    printf(" visitGlobalArea() str的地址是: %d\n",str);
    return str;
}

/*
访问堆区
*/
int* visitHeapArea(){
    printf("-----------------visitHeapArea()---------------------\n");
    int* chs = NULL;

    chs = (int*) malloc(sizeof(int)*10);
    for(int i=0; i<10; i++){
        *(chs+i) = i;
    }
    printf(" visitHeapArea() chs的地址是: %d\n",chs);
    return chs;
}
int main(){

    int a = visitStackArea();
    printf("main通过visitStackArea()得到a的值, a的地址是:  %d\n",&a);

    char* str = visitGlobalArea();
    printf("main通过visitGlobalArea()得到str的值, str的地址是: %d\n",str);

    int* chs = visitHeapArea();
    printf("main通过visitHeapArea()得到的chs的值, chs的地址是: %d\n",chs);
    free(chs);
    return 0;
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55

运行结果:

-----------------visitStackArea()---------------------
a的地址 3209584
main通过visitStackArea()得到a的值, a的地址是:  3209836
----------------visitGlobalArea()----------------------
 visitGlobalArea() str的地址是: 20470000
main通过visitGlobalArea()得到str的值, str的地址是: 20470000
-----------------visitHeapArea()---------------------
 visitHeapArea() chs的地址是: 7573184
main通过visitHeapArea()得到的chs的值, chs的地址是: 7573184
请按任意键继续. . .

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

结果分析: 
1,visitStackArea()函数内的地址是:3209584,而main函数调用visitStackArea()得到的a的地址是:3209836. 虽然main函数调用了函数visitStackArea(),但是 a是两块内存中的 . 当main调用visitStackArea()结束后, 编译器就把3209584地址释放了, 在main函数中重新开辟地址3209836来存储a

2,visitGlobalArea()函数内的str地址是:20470000, main函数调用visitGlobalArea()的地址也是20470000, 说明str的声明周期存在于整个程序的运行期间. 我们由上面可知, 他是存储在全局区的,当程序退出后由操作系统进行释放管理

3,visitHeapArea()函数内的chs地址是:7573184,main函数调用visitHeapArea()得到的chs地址也是:7573184,也说明chs声明周期比较长, 他声明周期长到何种程度,这有程序员来决定, 调用free方法来释放, 如果堆区的内存不手动free,那么将造成内存泄露(Memory Leak)

PS:假设有两个函数A,B, A调用B, 如果A主调函数想使用被调函数B中的变量分配的内存,可以吗?

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值