JVM
JVM的构成
类加载器(ClassLoader )
在JVM启动时或者在类运行时将需要的class加载到JVM中。
执行引擎:
负责执行class文件中包含的字节码指令;
内存区
也叫运行时数据区,是在JVM运行的时候操作所分配的内存区。运行时内存区主要可以划分为5个区域:
方法区(Method Area)
用于存储类结构信息的地方(类型信息是由类加载器在类加载时从类文件中提取出来的),包括常量池、静态变量、构造函数等。虽然JVM规范把方法区描述为堆的一个逻辑部分, 但它却有个别名non-heap(非堆)。方法区还包含一个运行时常量池。方法区和堆是被所有java线程共享的。
java堆(Heap)
存储java实例或者对象的地方。这块是GC的主要区域。方法区和堆是被所有java线程共享的。
java栈(Stack)
java栈总是和线程关联在一起,每当创建一个线程时,JVM就会为这个线程创建一个对应的java栈。在这个java栈中又会包含多个栈帧,每运行一个方法就创建一个栈帧,用于存储局部变量表、操作栈、方法返回值等。每一个方法从调用直至执行完成的过程,就对应一个栈帧在java栈中入栈到出栈的过程。所以java栈是线程私有的。
程序计数器(PC Register)
用于保存当前线程执行的内存地址。由于JVM程序是多线程执行的(线程轮流切换),所以为了保证线程切换回来后,还能恢复到原先状态,就需要一个独立的计数器,记录之前中断的地方,可见程序计数器也是线程私有的。
本地方法栈(Native Method Stack)
和java栈的作用差不多,只不过是为JVM使用到的native方法服务的。
本地方法接口
主要是调用C或C++实现的本地方法及返回结果。
GC机制
垃圾收集器一般必须完成两件事:检测出垃圾;回收垃圾。有以下几种方法检测出垃圾 :
引用计数法:
给一个对象添加引用计数器,每当有个地方引用它,计数器就加1;引用失效就减1。好了,问题来了,如果我有两个对象A和B,互相引用,除此之外,没有其他任何对象引用它们,实际上这两个对象已经无法访问,即是我们说的垃圾对象。但是互相引用,计数不为0,导致无法回收,所以还有另一种方法:
可达性分析算法
以根集对象为起始点进行搜索,如果有对象不可达的话,即是垃圾对象。这里的根集一般包括java栈中引用的对象、方法区常量池中引用的对象、本地方法中引用的对象等。
总之,JVM在做垃圾回收的时候,会检查堆中的所有对象是否会被这些根集对象引用,不能够被引用的对象就会被垃圾收集器回收。