Java内存划分和堆栈的简单整理

一直不理解堆栈的概念,很多视频课程讲解又不全面,学习代码的数据结构和算法成为了我最多困惑的地方。最近闲暇,查了下资料,基于Java语言整理下内存区域划分的基础知识。
由于Java程序是交由JVM执行的,所以在谈Java内存区域划分的时候事实上是指JVM内存区域划分。在讨论JVM内存区域划分之前,先来看一下Java程序具体执行的过程:


Java程序执行过程

运行时数据区模块

了解jvm的结构之前,有必要先来了解一下操作系统的内存基本结构


操作系统存储体系

以上是操作系统存储层次【CPU <--- > 寄存器<--- > 缓存(最多三级缓存)<--- >内存<--- >磁盘缓存<--- >固定磁盘存储<--- >可移动存储介质】的部分展示。
操作系统内存布局:
操作系统内部布局

操作系统内存的堆栈:

栈区(stack):由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈。
堆区 (heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表

栈是为执行线程留出的内存空间(当线程创建的时候,操作系统会为每个系统级的线程分配栈);
栈顶会为局部变量和数据预留块,当函数执行完毕,块就没有用了,可能在下次的函数调用的时候再被使用,栈通常都是采用后进先出的方式预留空间;因此最近的保留块通常最先被释放,从栈中释放块不过是指针的偏移(这里和数据结构中的栈的意思类似)

堆的数据结构并不是由系统支持的,而是由函数库提供的基本的malloc/realloc/free函数维护了一套内部的堆数据结构(所以才需要程序员自己释放内存,否则会造成内存泄漏);

堆包含了一个链表来维护已用和空闲的内存块;
申请内存:
当程序需要在堆中分配内存的时候,会从内部堆中寻找可用的内存空间,通过链表找到符合大小的内存块(链表遍历的方向是由低地址指向高地址),但是由于堆是不连续的内存区域,当找不到合适的内存区域的时候,则会利用系统调用来动态增加程序数据段的内存大小;
释放内存:
当系统受到程序的释放内存的申请的时候,会遍历该链表,寻找第一个空间大于所申请的堆结点,然后该结点会从链表中删除,并将该结点的空间释放给内存,这片内存空间又会返回到堆结构中,会经过内存块的组合,以便适合下次内存分配申请;(这里面如果没有管理内存分配在释放内存时很容易会造成内存碎片)

操作系统中的jvm

为什么jvm的内存是分布在操作系统的堆中呢??因为操作系统的栈是操作系统管理的,它随时会被回收,所以如果jvm放在栈中,那java的一个null对象就很难确定会被谁回收了,那gc的存在就一点意义都没有了,而要对栈做到自动释放也是jvm需要考虑的,所以放在堆中就最合适不过了。

操作系统+jvm的内存简单布局

JVM 的内存主要分为3个分区

堆区(Heap)-- 只存对象(数组)本身(引用类型的数据),不存基本类型和对象的引用。JVM只有一个堆区,这个“堆”是动态内存分配意义上的堆——用于管理动态生命周期的内存区域。JVM的堆被同一个JVM实例中的所有Java线程共享,它通常由某种自动内存管理机制所管理,这种机制通常叫做“垃圾回收”(garbage collection,GC)。JVM规范并不强制要求JVM实现采用哪种GC算法。
栈区(Stack)-- 栈中只保存基础数据类型的对象和对象引用。每个线程一个栈区,每个栈区中的数据都是私有的,其他栈不能访问。栈内有帧(方法调用会生成栈帧)分三个部分:基本类型变量区,执行环境上下文,操作指令区。
方法区 -- 又叫静态区,跟堆一样,被所有线程共享。方法区包含所有的class和static变量。方法区包含的都是在整个程序中永远唯一的元素。如:class,satic。


运行时jvm的处理

细化到增加jvm内部的处理和pc寄存器的配合时,可见,无论是在虚拟机中还是在我们虚拟机所寄宿的操作系统中功能目的是一致的,计算机上的pc寄存器是计算机上的硬件,本来就是属于计算机,计算机用pc寄存器来存放“伪指令”或地址,而相对于虚拟机,pc寄存器它表现为一块内存(一个字长,虚拟机要求字长最小为32位),虚拟机的pc寄存器的功能也是存放伪指令,更确切的说存放的是将要执行指令的地址,它甚至可以是操作系统指令的本地地址,当虚拟机正在执行的方法是一个本地方法的时候,jvm的pc寄存器存储的值是undefined,所以应该很明确的知道,虚拟机的pc寄存器是用于存放下一条将要执行的指令的地址(字节码流)。

当一个classLoder启动的时候,classLoader的生存地点在jvm中的堆,然后它会去主机硬盘上将A.class装载到jvm的方法区,方法区中的这个字节文件会被虚拟机拿来new A字节码(),然后在堆内存生成了一个A字节码的对象,然后A字节码这个内存文件有两个引用一个指向A的class对象,一个指向加载自己的classLoader,如下图。


image.png

方法区中的字节码内存块,除了记录一个class自己的class对象引用和一个加载自己的ClassLoader引用之外,还记录了以下信息。


image.png

从上面的图,不难发现,原来jvm的设计的模型其实就是操作系统的模型,基于操作系统的角度,jvm就是个java.exe/javaw.exe,也就是一个应用,而基于class文件来说,jvm就是个操作系统,而jvm的方法区,也就相当于操作系统的硬盘区,所以他也叫permanent区,因为这个单词是永久的意思,也就是永久区,我们的磁盘就是不断电的永久区。而java栈和操作系统栈是一致的,无论是生长方向还是管理的方式,至于堆,虽然概念上一致目标也一致,分配内存的方式也一直(new,或者malloc等等),但是由于他们的管理方式不同,jvm是gc回收,而操作系统是程序员手动释放,所以在算法上有很多的差异。

堆栈的概念有2种,通常除了内存上面的概念,还有数据结构里面的概念,两者不完全相同。简单的概念区分如下:

栈(操作系统-内存):由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈, 他们通常都是被调用时处于存储空间中,调用完毕立即释放
堆(操作系统-内存): 一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收,分配方式倒是类似于链表。生命周期由虚拟机的垃圾回收算法来决定(并不是一旦成为孤儿对象就能被回收)。所以调用这些对象的速度要相对来得低一些
堆(数据结构):堆可以被看成是一棵树,如:堆排序
栈(数据结构):一种后进先出的数据结构

参考资料:
JVM与操作系统
https://zhuanlan.zhihu.com/p/44401058
操作系统中堆和栈的区别
https://blog.csdn.net/SpeedMe/article/details/22943191
java之jvm学习笔记十三(jvm基本结构)
https://blog.csdn.net/yfqnihao/article/details/8289363
JVM的内存区域划分
https://www.cnblogs.com/dolphin0520/p/3613043.html
操作系统之堆和栈的区别
https://www.cnblogs.com/George1994/p/6399895.html
JVM学习(2)——技术文章里常说的堆,栈,堆栈到底是什么,从os的角度总结
https://www.cnblogs.com/kubixuesheng/p/5202561.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容