JVM
1. Java内存区域详解
线程私有:程序计数器,虚拟机栈,本地方法栈
线程共享的:堆,方法区,直接内存
1.1 各个区域详解
程序计数器
每个线程需要一个计数器记录自己执行到哪一行了。线程之间切换需要保存自己的执行位置,因此必须每个线程有自己的计数器。
生命周期随着线程创建而创建,随着线程结束而死亡。
java虚拟机栈
由一个个栈帧组成,每个栈帧都拥有局部变量表,操作数栈,动态链接,方法出口等信息。
局部变量表存放了编译期可知的各种数据类型,对象引用。
可能出现两种错误,StackOverFlowError
,OutofMemoryError
前者如果不允许虚拟机栈内存动态扩展,就可能因为递归导致栈深度超过当前java虚拟机最大深度的时候抛出异常。
后者是虚拟机栈内存大小可以动态扩展,如果虚拟机栈扩展着扩展着,没空间了,就会报错。
本地方法栈
很类似,但是执行的是本地方法。
堆
存放对象实例,之前是所有对象都存放在堆中,现在有些新技术的产生,栈上分配,导致不是所有对象都在堆中了。
如果某些方法中的对象引用没有被返回或者没有被外面使用,没有逃逸出去,那就直接在栈上分配内存。
垃圾回收主要在堆区域。可细分为老年代和新生代。Eden,FromSurvivor,ToSurvivor。
对象一般在Eden中分配,经历一次GC之后,eden和S0中存活的,放到是s1种,下次就是,eden和s1中存活的,放到s0中,并且每次的GC存活,年龄+1。超过阈值就放入老年代中。
方法区
(现在已经不在永久代,而是在元空间)存储已经被虚拟机加载的类信息,常量,静态变量等等。
常用参数:
-XX:MetaspaceSize=N//元空间的初始大小
-XX:MaxMetaspaceSize=M//最大大小