Java VM中经常被问到,也需要掌握的VM两个点,做一个小笔记
绝大多的言语都有共同的点,你会抄你同学的作业,那么你可能不知道你同学也会抄别人的作业,语言之间的底层构架也都差不多,Java底层的东西和其他语言也有类似,比如C/C++、phython、JavaScript等等
以前的JDK,VM内存可以分为三个点研究学习: 栈stack、堆heap、方法区method area
方法区method area存在堆中,是被认为比较特殊的存在,所以学习的时候会单拎知识点,有兴趣的小伙伴也会去oracle官网看一下“元空间”,以及本地内存的改动
栈
①表示方法执行的内存模型,每个方法被调用都会创建一个栈帧(局部变量、操作数、方法出口等)
②JVM为每个线程创建一个栈,用于存放该线程执行方法信息(实际参数、局部变量等)
③栈属于线程私有,不能实现线程间的共享
④栈的存储特性: 先进后出,后进先出
⑤栈是由系统自动分配,速度快,是一个连续的内存空间
方法执行,相关的调用都在栈中,每个方法被调用都会创建一个栈帧
如果在蓝色方法中再调用一个方法,会再生成一个新的栈帧,也一样,最后依据“先进后出,后进先出”的顺序,关闭执行完的方法,比如A方法中调用B方法,B方法调用C方法,那么顺序是A等B执行完,B等C执行完,C关闭=》B关闭=》A关闭
堆
①用于存放创建好的对象和数组
②JVM只有一个堆,被所有线程共享
③堆是一个不连续的内存空间,分配灵活,速度慢
new完的对象存在堆中
方法区 & 元空间
以往的JVM中也只有一个方法区,被所有线程共享,属于堆内的一部分,用于存储类、常量相关的信息,用来存放程序中永远不变或者唯一的内容,比如类信息[Class对象]、静态变量、字符串常量等
JDK7字面量(interned strings)、类的静态变量(class statics)转移到堆中,但是永久在还存在于中,直到JDK8,永久代才完全消失,转而使用元空间
元空间是直接存在内存中,不在java虚拟机堆中,因此元空间依赖于内存大小,可以自定义元空间大小,这里面存储的是类的元数据信息
类的元数据, 字符串池, 类的静态变量将会从永久代移除, 放入堆或者本地内存. 其中官方建议JVM的实现中将类的元数据放入 native memory, 将字符串池和类的静态变量放入堆中. 这样加载多少类的元数据就不在由MaxPermSize控制, 而由系统的实际可用空间来控制
减少OOM是一种连带原因, 更深层的原因还是要合并HotSpot和JRockit, JRockit没有永久代的东西, 运行状态比较好, 也不需要开发运维人员设置大小来调控
"java.lang.OutOfMemoryError: PermGen space ",这里的 “PermGen space”其实指的就是原来的方法区。不过方法区和“PermGen space”又有着本质的区别。前者是 JVM 的规范,而后者则是 JVM 规范的一种实现,并且只有 HotSpot 才有 “PermGen space”,而对于其他类型的虚拟机,如 JRockit(Oracle)、J9(IBM) 并没有“PermGen space”。由于方法区主要存储类的相关信息,所以对于动态生成类的情况比较容易出现永久代的内存溢出。最典型的场景就是,在 jsp 页面比较多的情况,容易出现永久代内存溢出
元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存,但可以通过以下参数来指定元空间的大小:
-XX:MetaspaceSize,初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过MaxMetaspaceSize时,适当提高该值。
-XX:MaxMetaspaceSize,最大空间,默认是没有限制的。除了上面两个指定大小的选项以外,还有两个与 GC 相关的属性:
-XX:MinMetaspaceFreeRatio,在GC之后,最小的Metaspace剩余空间容量的百分比,减少为分配空间所导致的垃圾收集 -XX:MaxMetaspaceFreeRatio,在GC之后,最大的Metaspace剩余空间容量的百分比,减少为释放空间所导致的垃圾收集----------------------------------------------------------------