-
JVM、JRE、JDK区别
JVM:JAVA虚拟机,JAVA源文件经过编译器编译的字节码交由JVM处理,负责跟操作系统交互,官方提供各个平台的JVM,JVM实现了JAVA的跨平台
JRE:JAVA运行时环境,包含JVM,一些库文件,以及运行JAVA所需的一些文件
JDK:JAVA开发框架,包含,编译器,调试器
-
JAVA内存分析
栈stack:方法执行的内存模型。栈是每个线程私有,不共享。栈由系统自动分配,速度快,栈是一个连续的内存空间
堆heap:存放创建好的对象和数组,jvm只有一个堆。被所有线程共享。堆是一个不连续的内存空间,分配灵活,速度慢
方法区method area:本身也是堆,用于存放类信息,静态常量,字符串常量等
-
垃圾回收机制 Garbage Collection (GC):发现无用对象并回收对象
3.1 垃圾回收算法
(1)引用计数法:被引用一次,则计数+1;被引用变量值,则计数-1,计数为0,则变为无用对象。缺点:循环引用时,对象引用次数永远不为0
(2)引用可达法:对象所有引用关系看作一张图,从GC Root开始,寻找该节点的引用节点,重复该操作,寻找完所有节点之后,未被引用的节点就是无用节点
3.2 垃圾分代回收(jvm将堆分成Eden、survivor、Tenured/Old)
(1)年轻代:新生成的对象会放在Eden区,年轻代的目标是快速收集掉生命周期短的对象,对应的Minor GC。Minor GC清理新生代内存时,采用复制算法将仍有用的对象移动到Survivor区
(2)年老代:在年轻代中经历了N(默认15)次GC之后还存活的对象,会被放到年老代中。年来带主要存放生命周期较长的对象,当年老代的对象越来越多时,会触发Major GC和Full GC
(3)持久代:用于存放静态文件、java类、方法等,持久带对垃圾回收没有显著影响
3.3 垃圾回收过程
(1)新创建的对象,绝大多数都会存储在Eden中
(2)当Eden满了(达到一定比例)不能创建新对象,则触发垃圾回收(GC),将无用对象清理掉,然后剩余对象复制到某个Survivor中,如S1,同时清空Eden区
(3)当Eden区再次满了,会将S1中的不能清空的对象存到另外一个Survivor中,如S2,同时将Eden区中的不能清空的对象,也复制到S1中,保证Eden和S1,均被清空。
(4)重复多次(默认15次)Survivor中没有被清理的对象,则会复制到老年代Old(Tenured)区中,
(5)当Old区满了,则会触发一个一次完整地垃圾回收(FullGC),之前新生代的垃圾回收称为(minorGC)
堆内存划分细节
3.4 JVM调优
对JVM调优,很大一部分工作就是对Full GC的调节,有如下几个原因可能导致Full GC:
(1)年老代(Tenured)被写满
(2)持久代(Perm)被写满
(3)System.gc()被显式调用(调有该函数只表示程序开发者建议系统启动垃圾回收,至于何时启动,由垃圾回收机制说了算)
-
JAVA类加载过程
4.1 加载(需要类加载器参与,类加载之后,会缓存到内存中)
(1)通过一个全限定类名,加载一个二进制流
(2)将二进制流中的静态存储结构转换为方法区的数据结构
(3)在堆中创建一个java.lang.Class对象,作为方法区数据的访问入口
4.2 链接
(1)验证:确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的
(2)准备:为静态变量在方法区创建内存,并使用默认初始值进行初始化
(3)解析:虚拟机将常量池的符号引用(以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能无歧义地定位到目标即可)转换为直接引用(可以是直接指向目标的指针、相对偏移量或是一个能间接定位到目标的句柄)
4.3 初始化
(1)执行类的构造器方法<client>()的过程:<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的
(2)若初始化一个类时,其父类没有初始化,则先执行父类的初始化