文章目录
- 博客概述
- 堆,栈,方法区概念和联系
- 辨清java堆
- java栈
- 方法区
博客概述
在这篇博客里面主要介绍关于堆,栈,方法区的相关内容,针对jdk1.7
堆,栈,方法区概念和联系
堆解决的是数据存储的问题,即数据怎么放,放在哪?
栈解决的是程序的运行问题,即程序如何执行,或者说如何处理数据。
自己理解:堆放真正的数据对象,栈放引用(理解为一个指针)
方法区则是辅助堆栈的快永久区(Perm),解决堆栈信息的产生,是先决条件。
我们创建一个新的对象,User:那么User类的一些信息(类信息,静态信息都存在于方法区中,可以理解为一个模板)
而User类被实例化出来之后,被存储到java堆中的一块内存空间。当我们去使用的时候,都是使用user对象的引用,形如:User u = new user();这里的u就是放在java栈中的,也就是User真实对象的一个引用(指针)。
辨清java堆
java堆是和java应用程序关系最密切堆内存空间,几乎所有堆对象都存放其中,并且java堆完全是自动化管理的,通过垃圾回收机制,垃圾对象会自动回收清理,不需要显示的释放。
根据垃圾回收机制的不同,java堆可能拥有不同堆结构。最为常见的就是将整个java堆分为新生代和老年代。其中新生代中存放新生的对象或者年龄不大的对象,老年代则存放老年对象。
新生代分为eden区(伊甸园),s0区,s1区,这两个区域也被称为from区域和to区域,他们是两块大小相等并且可以互换角色的空间,至于为什么这么说跟这两个区域的垃圾回收算法有关系。
绝大多数情况下,对象首先分配在eden区域,在一次新生代回收之后,如果对象还存活,则会进入s0或者s1区域,之后每经过一次新生代的回收,如果对象存活则他的年龄就加1,当对象达到一定年龄之后就进入老年代。
个人理解:刚实例化出来的对象基本都在eden区域。s0和s1大小相等,可以互换角色的空间。不能同时使用是因为复制算法的存在。简单介绍一下复制算法,如下。
简单的理解一下,这种算法的好处就类似于格式化,肯定是比一个一个寻址找到具体的对象然后删除要快,而且不会产生内存碎片。gc对新生代频繁回收,老年代里的对象经历了很多次的gc,回收比较少。
java栈
java栈是一块线程私有的内存空间,一个栈,一般由三部分组成:局部变量表,操作数栈和帧数据区。
- 局部变量表:用于报错函数的参数及局部变量
- 操作数栈:主要用于保存计算过程的中间结果,同时作为计算过程中变量的临时的存储空间。(局部变量,临时变量就存在这)
- 帧数据区:除了上述两者之外,栈还需要一些数据来支持常量池的解析,这里的帧数据区保存着访问常量池的指针,方便程序访问常量池,另外,当函数返回或者出现异常时,虚拟机必须有一个异常处理表,方便发送异常的时候找到异常的代码,因此异常处理表也是帧数据区的一部分。
方法区
java方法区和堆一样,方法区是一块所有线程共享的内存区域,它保存系统的类信息,比如类的字段,方法,常量池等。方法区的大小决定了系统可以保存多少个类,如果系统定义太多的类,就会导致方法区溢出。虚拟机同样会抛出内存溢出错误。方法区可以理解为永久区(Perm)。要灵活根据项目的规模决定方法区域的大小。