运行时数据区域
1.方法区 non-heap(非堆)
- 各线程共享的区域(为什么说是各线程共享)
- 用于存放虚拟机加载的类信息,常量信息,静态变量,即时编译器编译的代码等(为啥这么说,哪里有资料可以查)
- 并不能完全等同于永久代(Permanent generation)(永久代的主要特性是什么,和方法区的区别是什么)
- 垃圾回收在这里比较少见,回收目标是常量池和对类型的卸载
2.虚拟机栈:线程私有的,生命周期和线程相同(就是说一个线程启动就有一个虚拟机栈)
*描述的是java方法执行的内存模型。每个方法在执行时都会创建一个栈帧,用来记录局部变量,动态链接,方法出口等,每个方法的执行从开始到介绍就是一个栈帧从虚拟机入栈道出栈的过程
什么是栈帧?可以理解为一个方法的运行空间,它主要有两部分构成,一部分是局部变量表,方法中定义的局部变量以及方法的参数就是放在这张表中;另一部分是操作数栈,用来存放操作数(什么是操作数?)
-
这个区域规定了两种异常
- 如果线程请求的栈深度大于虚拟机允许的栈深度,则抛出stackOverFlowError,比如递归调用
- 如果虚拟机栈可以动态扩展,但是扩展时申请不到足够的内存则抛出OutOfMemoryError,比如这个线程运行时创建大量的类
3.本地方法栈
- 本地方法栈和虚拟机栈类似。区别是虚拟机栈为执行java方法服务,本地方法栈为运行native服务
4.java堆
- 是java虚拟机中内存区域最大的一块(为甚会是最大的一块)
- 是被所有线程共享的区域,虚拟机启动时创建(为什么要被线程共享)
- 几乎所有的对象实例都会在这分配空间(怎么确定分配多少空间)
- 可以是物理上不连续的区域,只要逻辑上连续即可(物理不连续,逻辑连续怎么理解)
- 如果堆中没有内存可以分配,并且不能扩展的话,抛出OutOfMemory
5.计数器
- 可以看成是当前线程所执行字节码的行号指示器
- 每个线程都需要一个独立的程序计数器,所以是私有的。(java虚拟机多线程是通过轮流切换并分配处理器时间的方式)
- 如果执行的是java方法,计数器记录的是字节指令地址;如果是native方法,计数器为空(uundifined),是唯一没有outOfMemory的区域
native是本地方法,和平台有关需要借助c语言
6.运行时常量池,是方法区的一部分。
- class文件除了记录类的版本,字段,方法等,还有常量池,用于存放编译期生成的字面量和字符号引用,在累的加载后进入该区域。
- 具有动态性,运行期间也可以将行的常量放入池中。比如String类中的intern()方法
7.直接内存:不属于虚拟机内存,但是有可能导致outOfmemoryError异常
方法区
为什么说是各线程共享?
1.我们已经知道方法区是存放加载的类信息,常量,静态变量,即时编译的代码等
2.java是编译型语言,编译之后不会改变
方法区是存放加载的类信息,常量,静态变量,即时编译的代码为啥这么说,哪里有资料可以查。
资料没有找到,但是可以看戏这篇文章
https://www.zhihu.com/question/23599282
堆
永久代的主要特性是什么,和方法区的区别是什
java虚拟机分为3个区,young新生代,old老年代和permanernt
Young保存刚实例化的对象,当区域被填满时,GC会将对象移动到Old区,Permanent区保护反射对象
堆内存分配:
VM初始分配的堆内存由-Xms指定,默认是物理内存的1/64;
JVM最大分配的堆内存由-Xmx指定,默认是物理内存的1/4。
默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx的最大限制;
空余堆内存大于70%时,JVM会减少堆直到-Xms的最小限制。
因此服务器一般设置-Xms、-Xmx 相等以避免在每次GC 后调整堆的大小。
说明:如果-Xmx 不指定或者指定偏小,应用可能会导致java.lang.OutOfMemory错误,此错误来自JVM,不是Throwable的,无法用try…catch捕捉。
java.lang.OutOfMemoryError: PermGen space。
内存溢出是不能捕获的
java进程运行过程中创建的对象存放在堆中,堆被划分成两个不同的区域:新生代 ( Young )、老年代 ( Old )。新生代 ( Young ) 又被划分为三个区域:Eden、From Survivor、To Survivor。
申请一块内存的过程
1.jvm试图为相关java对象在eden中初始化一块内存区域
2.当eden空间足够时,内存申请结束,否则到下一步
3.jvm试图释放eden中所有不活跃的对象;释放后若eden空间仍让不足已放入新对象,则试图将部分eden(伊甸园)中活跃对象放入survivor(幸存者)区/old区
4.Surivor区被用来作为eden及old的中间交换区,当old区空间足够时,survivor区对象被移动到old区,否则被保留在survivor区
5.当old去不够时,jvm会在old区进行完全的垃圾收集
6.完全垃圾收集够,若survivor及old区仍然无法存放Eden复制过来的部分对象向,导致jvm无法再eden区为新对象创建内存区域,则出现“out of momory错误”