C++的内存有五大分区:堆区、栈区、自由存储区、全局/静态存储区、常量存储区。

     五个数据段:数据段、代码段、BSS段、堆、栈

内存分配方式有三种:

  1. 从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在。例如全局变量,static变量。
  2. 在栈上创建。 在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 这个大小在linux上可以使用:ulimit -a | grep stack 来查看
  3. 从堆上分配,亦称动态内存分配。程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用非常灵活,但问题也最多。
 
#include <iostream>
using namespace std;
char * GetMemory(void){
    //p_arr是个数组的起始地址,这个数组的内容是存放在栈上面的。另外,非常值得一
    //提的是:系统没有为p_arr单独分配一个指针变量用来存放数组p_arr的起始地址。 
    char p_arr[] = "p_arr : hello world";     //p_heap是一个指针变量,这个指针变量存在于栈上面,指针变量的值是指向的由
    //malloc分配得到的堆上的内存空间的起始地址。
    char *p_heap=(char *)malloc(sizeof(char)*20);     //ptr是一个指针变量,这个指针变量存在于栈上面,指针变量的值是指向的后面字符
    //串的起始地址,这个字符串是常量,是存在于静态存储区域的,不是在栈上,也不是
    //在堆上。
    char *ptr = "ptr : hello world";     strcpy(p_heap,"p_heap: hello world");
    printf("%08X\n",p_arr);
    printf("%08X\n",&p_arr);
    printf("%08X\n",p_heap);
    printf("%08X\n",&p_heap);
    printf("%08X\n",ptr);
    printf("%08X\n",&ptr);
    puts(p_arr);
    puts(ptr);
    puts(p_heap);
    //return p_arr;
    return ptr;
}
int main(){
    char * str=NULL;
    str = GetMemory();
    printf("%08X\n",str);
    printf("%08X\n",&str);
    puts(str);
    return 0;
}  

上面程序输出的结果是:

DC726BF0 
DC726BF0
0EB29010
DC726BE8
004009D8
DC726BE0
p_arr : hello world
ptr : hello world
p_heap: hello world
004009D8
DC726C28
ptr : hello world
 

这里面8有8行关于地址的输出,分别描述如下:

  • DC726BF0这个地址是栈上面的,函数退出后,这块地址空间就废弃了。
  • DC726BF0是栈上面的地址,之所以和上面一样,是因为栈上并没有分配一个单元用来存放数组指针,这个是使用char []和char * 的最根本的区别。
  • 0EB29010这个地址是在堆上面的,直到delete才会释放掉这块内存。
  • DC726BE8是栈上面的地址,用来存放指针变量p_heap的单元,p_heap的值就是上面堆上内存的地址。
  • 004009D8这个地址是静态存储区的,函数退出后不会清除掉
  • DC726BE0栈上面的地址,用来存放指针变量ptr的单元,ptr的值就是上面静态的存储区地址
  • 004009D8这个是由函数返回的地址,就是上面静态存储区的地址
  • DC726C28这个地址是main函数局部变量的存放地址,位置要比栈上的位置低
从这个例子上也可以看出,栈的地址空间、堆的地址空间、静态存储空间、main函数的局部地址空间的关系,地址位置从高到低关系如下:

栈-->main函数局部变量地址-->堆-->静态变量地址

可以看出来,静态地址最低,然后依次是:堆、main、栈。

另外的,还有一个“先声明的变量时在高地址区还是在低地址区”的问题,这个问题要看栈怎么生长的了,如果栈底在高内存,向低地址生长,自然先声明的变量在高地址。例如:在linux中,栈底就是在高地址的,所以后面声明的变量地址就是低地址。此外,在堆上的地址一般是从低到高的。下面的例子可以清晰的看出这两种情况。

 
#include <iostream>
class c1
{
    public:
        int number;
};
main()
{
    int a =3;
    int b =4;
    printf("addr of a on stack: 0x%X\n",&a);
    printf("addr of b on stack: 0x%X\n",&b);
    c1 * c=new c1();
    c1 * d=new c1();
    int * e =new int(100);
    printf("addr of c on stack: 0x%X--0x%X(on heap)\n",&c,c);
    printf("addr of d on stack: 0x%X--0x%X(on heap)\n",&d,d);

 

以上代码的输出:

addr of a on stack: 0x26F8A744
addr of b on stack: 0x26F8A740
addr of c on stack: 0x26F8A738--0xE7F1010(on heap)
addr of d on stack: 0x26F8A730--0xE7F1030(on heap)

对于上面的输出解释如下:

  • a,b是位于main函数栈上的变量,他们地址的增长方向是从高到低的。另外,也请注意a和b的地址相差的距离是4个字节(虽然是64位机器和OS,但是int占用的空间还是32位)
  • c,d是位于main函数栈上的指针变量,同样是从高到低增长。这次要注意,他们相差的距离是8个字节(因为是c和d都是指针变量,要能够存下64位地址长度的数据)
  • 最后,要注意c和d指向的结构都位于堆上,这次他们地址的增长方向就是从低到高(从0xE7F1010到0xE7F1030)了。

转载地址:http://www.cnblogs.com/welkinwalker/archive/2011/01/05/1926203.html

从一个微型例子看“C/C++的内存分配机制”和“数组变量名与指针变量名”(转)的更多相关文章

  1. C语言定义一个指针变量

    10.2.1 定义一个指针变量 对指针变量的定义包括三个内容: (1) 指针类型说明,即定义变量为一个指针变量: (2) 指针变量名: (3) 变量值(指针)所指向的变量的数据类型. 其一般形式为: ...

  2. Spark小课堂Week7 从Spark中一个例子看面向对象设计

    Spark小课堂Week7 从Spark中一个例子看面向对象设计 今天我们讨论了个问题,来设计一个Spark中的常用功能. 功能描述:数据源是一切处理的源头,这次要实现下加载数据源的方法load() ...

  3. 一个例子看懂所有nodejs的官方网络demo

    今天看群里有人用AI技术写了个五子棋,正好用的socket.io,本身我自己很久没看nodejs了,再加上Tcp/IP的知识一直很弱,我就去官网看了下net.socket 发现之前以为懂的一个官方例子 ...

  4. SQL Server Reporting Service(SSRS) 第一篇 我的第一个SSRS例子

    很早就知道SQL SERVER自带的报表工具SSRS,但一直没有用过,最近终于需要在工作中一展身手了,于是我特地按照自己的理解做了以下总结: 1. 安装软件结构 SSRS全称SQL Server Re ...

  5. 一个简单例子:贫血模型or领域模型

    转:一个简单例子:贫血模型or领域模型 贫血模型 我们首先用贫血模型来实现.所谓贫血模型就是模型对象之间存在完整的关联(可能存在多余的关联),但是对象除了get和set方外外几乎就没有其它的方法,整个 ...

  6. 一个经典例子让你彻彻底底理解java回调机制

    转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273),请尊重他人的辛勤劳动成果,谢谢 所谓回调: ...

  7. 回调--一个经典例子让你彻彻底底理解java回调机制

    本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17483273),请尊重他人的辛勤劳动成果,谢谢 以前不理解什么叫回调 ...

  8. Lea指令计算地址(用于四则混合运算),附上一个函数调用例子及其反汇编代码,很清楚

    比如你用local在栈上定义了一个局部变量LocalVar,你知道实际的指令是什么么?一般都差不多像下面的样子:     push   ebp     mov   esp,   ebp     sub ...

  9. 用 Lua 实现一个微型虚拟机-基本篇

    用 Lua 实现一个微型虚拟机-基本篇 目录 介绍 机器指令模拟 最终核心代码 虚拟机内部状态可视化 完整项目代码 后续计划 参考 介绍 在网上看到一篇文章 使用 C 语言实现一个虚拟机, 这里是他的 ...

随机推荐

  1. Linux进程实践(4) --wait避免僵尸进程

    Wait的背景 当子进程退出的时候,内核会向父进程发送SIGCHLD信号,子进程的退出是个异步事件(子进程可以在父进程运行的任何时刻终止) 子进程退出时,内核将子进程置为僵尸状态,这个进程称为僵尸进程 ...

  2. C语言的指针的基本语法

    这是我在C++编程的一本教科书偶然看到的,就当做一个查询的资料吧

  3. 《java入门第一季》之StringBuffer小案例

    这里是针对其反转功能来举的例子,再对比之前写的一篇String类的反转功能,StringBuffer明显提高了代码量,提高了效率. import java.util.Scanner; /* * 把字符 ...

  4. SpriteBuilder中子节点的相对位置(%百分比定位)

    子节点(或在这里确切的为精灵sprites)50%的偏移效果使得其在父节点中居中显示,该父节点的纹理在左下角(锚点为0,0). 这样做好过用父节点的位置的实际值来定位.根据父节点实际位置来定位在早期的 ...

  5. Linux - 延伸正则表达式

    RE 字符 意义与范例 + 意义:重复『一个或一个以上』的前一个 RE 字符 范例:搜寻 (god) (good) (goood)... 等等的字串. 那个 o+ 代表『一个以上的 o 』所以,底下的 ...

  6. Mahout SlopOne

    关于推荐引擎 如今的互联网中,无论是电子商务还是社交网络,对数据挖掘的需求都越来越大了,而推荐引擎正是数据挖掘完美体现:通过分析用户历史行为,将他可能喜欢内容推送给他,能产生相当好的用户体验,这就是推 ...

  7. VS2010 / MFC + OpenCV 2.4.9打开图片

    原文地址:http://www.opencv.org.cn/forum.php?mod=viewthread&tid=30832 第一部分,参考http://jingyan.baidu.com ...

  8. HBase 二级索引与Join

    二级索引与索引Join是Online业务系统要求存储引擎提供的基本特性.RDBMS支持得比较好,NOSQL阵营也在摸索着符合自身特点的最佳解决方案. 这篇文章会以HBase做为对象来探讨如何基于Hba ...

  9. PHP开发环境apache搭建

    首先我们先来了解一下PHP的一些相关的基础知识: PHP是啥? php其实就是超文本预处理程序,一种制作网站的脚本程序. 通常PHP的运行环境有以下两种: wamp  windows+apache+m ...

  10. 快速熟悉Oracle索引

    一.索引 1.1 什么是索引? 一种用于提升查询效率的数据库对象: 通过快速定位数据的方法,减少磁盘的输入输出操作: 索引信息与表独立存放: Oracle数据库自动使用和维护索引. 1.2 索引分类 ...