一个程序的运行是需要内存的,那么我们平常写的程序的内存都是怎么分配的呢

(1)首先我们要知道,内存是真实存在的,内存是一个物理器件。它时由操作系统管理的,我们平常只要使用它就行了,为了方便管理。操作系统提供了很多种机制来管理内存,每一种机制都有其特点。
(2)三种内存来源:栈(stack)、堆(heap)、数据区(data)
(3)我们获取内存的来源就是其中一种,(C语言程序中)

1.什么是栈?

1.1.栈的特点
(1)使用时自动分配,用完自动释放。

  1. #include<stdio.h>
  2. main()
  3. {
  4. int a = ;
  5. int b = ;
  6.  
  7. }

总结:a和b都是局部变量,都是栈内存,自动分配和释放

(2)栈内存反复使用。使用完后不会清零,这就是为什么每次分配局部变量若不初始化就是随机值的原因。

  1. int a ;
  2. printf("a = %d\n",a);
  3. // a是随机值

(3)临时性:函数不能返回栈变量的指针。局部变量的地址是不能返回的。在一个函数内部分配的局部变量在函数结束时,其内存就已经不存在了。返回其地址是很危险的。

  1. #include<stdio.h>
  2. int * test();
  3. main()
  4. {
  5. int * w; //局部变量,栈上分配
  6. w = test();
  7. *w = ;
  8. }
  9.  
  10. int* test()
  11. {
  12. int a = ; //局部变量,栈上分配 局部变量,作用域为test函数,生命周期为test函数从执行到执行完毕
  13. int *p = &a;
  14. return p;
  15. }

分析:

(1)test函数能不能返回p的值?

test函数中的a和p都是局部变量,作用域为test函数,生命周期为test函数从执行到执行完毕,当test函数执行完后,为a和p分配的栈上的内存已经释放了。也就是说,原来分配的地址已经不能在被a和p使用了。这里可以返回p的值,只是没有意义了,而是很危险的行为。
(2)能不能通过 *w = 20?

能访问,但可能会出错,原a的内存已经被释放,原来的地址已经不再属于a,这时候就会使程序崩溃。访问了不属于它的地址。
(3)数组越界问题 (栈溢出)

  1. main()
  2. {
  3. int arr[]; //局部变量,栈上分配
  4.  
  5. arr[] = ;
  6. }

给arr数组分配了5个地址空间,访问第11个空间则会出错。

2.什么是堆?

(1)操作系统堆管理器管理。堆管理器是操作系统的一个模块
(2)大块内存,分配自如。按需分配。
(3)堆内存手动分配&释放,malloc,free
(4)堆内存也是反复使用的, 堆管理器不会去清零堆内存,下一个使用的变量才去清零。
(5)堆内存。从分配到释放一直属于同一个进程,之前或者之后都不可以再访问。

  1. //堆内存的使用范例
  2.  
  3. main()
  4. {
  5. int*p=(int *)malloc(*sizeof(int));
  6. }
  7. //使用malloc()分配的内存是在堆上分配的
  8. //这里的p仍旧是栈上分配,p是一个指针。该指针指向的内存是堆内存。(4000个字节的首四个字节)

3.数据区

(1)数据段:(也被称作数据区、静态数据区、静态区)程序运行所需要的数据存放在这,比如函数执行过程中调用的一些变量(全局变量),产生的一些数据。注意:全局变量才算是程序的数据,局部变量算是函数的数据,局部变量不是程序的数据。 局部变量属于栈管理,在栈分配。
(2)bss段
bss段:(也叫ZI段,zero initial 段,bss段的特点就是被初始化为0,bss段本质上也是数据段,bss段就是被初始化为0的数据段)

总结:(1)全局变量初始化为非0,存放在数据段。
static修饰的局部变量存放到数据段
(2)全局变量 未初始化 或 初始化为0,存放在bss段
(3)bss段是数据区的一部分

4.代码段

就是存放代码的地方,需要注意的是像char *p="aabbcc";这样的也会被分配的代码段,代码段的东西时不可以被修改的。

接下来我们看一个程序,具体分析,每一个变量存放在哪里

  1. #include<stdio.h>
  2.  
  3. int test1 = //全局变量,数据区
  4. int test2 = //初始化为0的全局变量,数据区的bss段
  5. int test3 //未初始化全局变量,数据区的bss段
  6.  
  7. char * display();//代码段
  8.  
  9. void main()
  10. {
  11.  
  12. int ok1 = ; /* 局部变量
  13. int ok2 = 0; 栈上分配
  14. int ok3; */
  15.  
  16. int * pre; //局部变量指针,栈上分配
  17. pre = display();
  18.  
  19. }
  20.  
  21. int * display();
  22. {
  23. char * word = "he is goudan";
  24. //代码段
  25.  
  26. int * re = (int*)malloc(*sizeof(int));
  27. //re指针在栈分配
  28. //re指针指向的是内存在堆上分配
  29. //当display()函数执行完毕后,re指针本身地址被释放,
  30. //re所指向的堆内存依旧存在(若未保留该指针则内存泄漏)
  31. return re;
  32. }

总结:

(1)程序经过编译后,分成不同的段,程序就是由好多个段组成的。数据段,代码段,bss段。
(2)全局变量才属于程序的数据,局部变量属于函数的数据。
(3)有些特殊数据会放在代码段,像字符串。

堆、栈、数据区、bss、代码段的更多相关文章

  1. PHP 堆 栈 数据段 代码段 存储的理解

    对象在PHP里面和整型.浮点型一样,也是一种数据类,都是存储不同类型数据用的, 在运行的时候都要加载到内存中去用,那么对象在内存里面是怎么体现的呢? 内存从逻辑上说大体上是分为4段,栈空间段.堆空间段 ...

  2. 内存布局:栈,堆,BSS段(静态区),代码段,数据段

    简介 我们程序运行的时候都是放在内存里的.根据静态.成员函数.代码段.对象.等等.放在不同的内存分块里.大概分为5块 1  栈 2  堆 3 BSS段-全局区-(静态区) 4 代码段 5 数据段 栈 ...

  3. java虚拟机 jvm 栈数据区

    java栈帧还是需要一些数据支持常量池的解析.正常方法的返回和异常的处理.大部分的java字节码指令需要进行常量池的访问,在栈帧数据区中保存着访问常量池的指针,方便程序访问java常量池.如下图所示: ...

  4. Java 底层机制(JVM/堆/栈/方法区/GC/类加载)

    转载:https://www.jianshu.com/p/ae97b692614e?from=timeline JVM体系结构 JVM是一种解释执行class文件的规范技术.   JVM体系结构 我翻 ...

  5. JVM堆 栈 方法区详解

    一.栈 每当启用一个线程时,JVM就为他分配一个JAVA栈,栈是以帧为单位保存当前线程的运行状态 栈是由栈帧组成,每当线程调用一个java方法时,JVM就会在该线程对应的栈中压入一个帧 只有在调用一个 ...

  6. java 堆 栈 方法区的简单分析

    Java里的堆(heap)栈(stack)和方法区(method) 基础数据类型直接在栈空间分配, 方法的形式参数,直接在栈空间分配,当方法调用完成后从栈空间回收.   引用数据类型,需要用new来创 ...

  7. linux进程的堆栈空间_代码段(指令,只读)、数据段(静态变量,全局变量)、堆栈段(局部变量)、栈【转】

    转自:http://blog.csdn.net/gongweijiao/article/details/8207333 原文参见:http://blog.163.com/xychenbaihu@yea ...

  8. vs中的强大的代码段管理

    vs中的代码段管理可以实现大段固定文本的快捷输入,方法: 首先编写.snippet文件如: <?xml version="1.0" encoding="utf-8& ...

  9. linux代码段,数据段,BSS段, 堆,栈(二)

    //main.cpp int a = 0; 全局初始化区  char *p1; 全局未初始化区 main() { int b; 栈 char s[] = "abc"; 栈 char ...

随机推荐

  1. 【HIVE】(1)建表、导入数据、外部表、导出数据

    导入数据 1). 本地 load data local inpath "/root/example/hive/data/dept.txt" into table dept; 2). ...

  2. MAVEN添加本地仓库和注意事项!

    将jer包加载本地仓库导命令 注意:电脑配置了maven的环境变量, 安装指定文件到本地仓库命令:mvn install:install-file -Dfile=       : 指定jar文件路径与 ...

  3. Java实现 LeetCode 834 树中距离之和(DFS+分析)

    834. 树中距离之和 给定一个无向.连通的树.树中有 N 个标记为 0-N-1 的节点以及 N-1 条边 . 第 i 条边连接节点 edges[i][0] 和 edges[i][1] . 返回一个表 ...

  4. Java实现 蓝桥杯 算法训练 Pollution Solution

    试题 算法训练 Pollution Solution 问题描述 作为水污染管理部门的一名雇员,你需要监控那些被有意无意倒入河流.湖泊和海洋的污染物.你的其中一项工作就是估计污染物对不同的水生态系统(珊 ...

  5. Java实现 LeetCode 669 修剪二叉搜索树(遍历树)

    669. 修剪二叉搜索树 给定一个二叉搜索树,同时给定最小边界L 和最大边界 R.通过修剪二叉搜索树,使得所有节点的值在[L, R]中 (R>=L) .你可能需要改变树的根节点,所以结果应当返回 ...

  6. 用斗地主的实例学会使用java Collections工具类

    目录 一.背景 二.概念 1.定义 2.方法 2.1.排序方法 2.2.查找/替换方法 三.斗地主实例 3.1.代码结构 3.2.常量定义 3.3.单只牌类 3.4.玩家类 3.5.主程序 四.深入理 ...

  7. 快速幂解法--x^n

    class Solution{ public: double myPow(double x,int n){ if(==x || n==) return ; if(n == ) return x; if ...

  8. (一)linux三剑客之grep

    给自己提出以下6个问题,看自己是否真正掌握了grep [1] grep 是什么? [2] grep 有什么作用 ? [4] grep 常用于何处 ? [5] grep 的基本用法 ? [6] grep ...

  9. Js中Array 函数使用方法

    遇到数组有关操作,脑子第一反应不要再是嵌套 for 循环了,Array 类型提供了一些遍历有关的函数. Array.prototype.forEach() : 把数组每个元素丢到一个处理 functi ...

  10. @atcoder - CODE FESTIVAL 2017 Elimination Tournament Round 3 F@ Unicyclic Graph Counting

    目录 @description@ @solution@ @accpeted code@ @details@ @description@ 求有多少 n 点 n 边的无向连通图,满足第 i 个点的度数为 ...