因为最近在复习java基础,所以写一点java后台相关的帖子啦~
java内存模型
指的是jvm工作内存的划分和结构。对于这块有了解的的话在实际生产中对于调整虚拟机参数,或者遇到OOM等error时知道如何排查问题。
java内存模型图
具体分为两部分,一部分所有线程共享的,包括方法区,堆(heap)。
另一部分是线程独有的,包括堆栈(分为native方法栈,虚拟机方法栈)和程序计数器。
-
方法区
也叫永久代,可以设置jvm参数来决定GC是否回收方法区。主要用来存储类中例如类信息、常量、静态变量、即时编译器编译后的代码等数据。对于这块的垃圾回收主要针对常量和类型的卸载。常量分为字面量和符号引用。字面量如文本字符串、被声明为final的常量值等。而符号引用则属于编译原理方面的概念,包括下面三类常量:
类和接口的全限定名
字段的名称和描述符
方法的名称和描述符
判定为无用类的条件也比较苛刻,需要在堆中没有这个类的任意实例,没有使用反射获取此类,此类的classloader已经被虚拟机卸载。
从上面来看,permgen的回收效率很低。但是如果对于那种能动态生成很多类的框架,比如jsp等,需要回收类型以防方法区溢出。 -
堆
分为老年代和新生代,GC主要就是在堆上进行的。分代垃圾收集机制提升了回收效率。因为针对不同堆区域的对象特点使用了不同的算法。young:old 默认是1:2空间。回收算法是stop the world(目的是更加高效)。当然也有各种不同的虚拟机,会有一些不同,例如有种虚拟机不用stop the world,具体可以去查看周志明的java虚拟机原理那本书讲的很详细。
youngGen:分为三部分:Eden,from survivor,to survivor。默认内存分配比例是8:1:1。当然可以在jvm参数中设置这个比例。在youngGen中使用的是复制算法。在youngGen中进行的是minor GC,每次只预留一个survivor,GC时将另外区域的存活对象复制过来。并且将存活的对象的年纪加1.下一次使用这个survivor与空Eden作为分配新对象的空间。当一个survivor不够存放所有对象,或者有对象年纪超过阈值,会由minor触发full GC。
OldGen:老年代使用的是标记-清除算法。标记出回收的内存区域。这种算法会造成很多内存碎片。而分配对象需要连续的内存区域。full GC会比minor更加耗时和资源。 -
程序计数器
这里可以理解为指向字节码执行的对应行的指针。 -
虚拟机方法栈
这个是线程运行虚拟机中方法所分配的内存空间。虚拟机中的方法指的是字节码中对应的在运行时调用的方法。 -
本地方法栈
指的是一些native方法的调用所占用的内存空间。是指jvm调用的平台层面的c/c++代码(方便jvm的跨平台)。
当然随着jvm版本的优化还有直接内存的使用。也就是为了避免变量在栈之间多次复制(比如说方法之间的变量值传递)。逃逸技术相关的栈上直接分配内存,指的是有些对象直接在栈上分配而不是堆里。更多详细信息可以在网上查阅关键字。