JVM 虚拟机主要涉及两个概念
1, JVM内存结构
2, JVM类的加载
JVM的内存结构
可以分为栈, 堆, 方法区。
栈: 存放局部变量(基本数据类型, 对象引用) StackOverflowError
堆: 存放对象实例。 又称GC堆。 OutOfMemoryError Java Heap Space
-Xms512m -Xmx1024m
方法区: 存放Class和Meta信息。 运行时常量。 又称永久代,PermantGeneration, 不被GC收集。
OutOfMemoryError PermGen Space
新生代Eden 老年代Old Survivor1 Survivor2
新生代GC(Minor GC) 速度快
老年代GC(Major GC / Full GC) 速度慢
大对象/ 长期存活对象直接进入老年代
一, 类的加载, 连接, 初始化
•加载:查找并加载类的二进制数据
•连接
–验证:确保被加载的类的正确性
–准备:为类的静态变量分配内存,并将其初始化为
默认值
–解析:把类中的符号引用转换为直接引用
•初始化:为类的静态变量赋予
正确的初始值
可以用static块表示
package com;
class S{
private static S s = new S();
public static int count1;
public static int count2=0;
private S(){
count1++;
count2++;
}
public static S getS(){
return s;
}
}
public class Test1 {
public static void main(String[] args) {
S s = S.getS();
System.out.println("count1: " + s.count1); // count1: 1
System.out.println("count2: " + s.count2); //count2: 0
}
}
•Java程序对类的使用方式可分为两种
–主动使用
–被动使用
•所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们
•主动使用(六种)
–创建类的实例
–访问某个类或接口的静态变量,或者对该静态变量赋值
–调用类的静态方法
–反射(如Class.forName(“com.shengsiyuan.Test”))
–初始化一个类的子类
–Java虚拟机启动时被标明为启动类的类(Java Test)
•除了以上六种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化
典型的是访问某个类的静态常量 用final修饰的
package com;
import java.util.Random;
class F{
// 6/3 在编译阶段就能算出, 反编辑class文件,结果是2
// 如果不加final就不是常量,会先输出static block
// 如果x=new Random().nextInt(100); 即要运行阶段才能得到的值,会输出static block
public static final int x = 6/3;
static {
System.out.println("static block");
}
}
public class Test2 {
public static void main(String[] args) {
System.out.println(F.x); //2
}
}
•类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构
•类的加载的最终产品是位于堆区中的Class对象
•Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口
二, 自定义类加载器
用用户自定义的类加载是可以被卸载的。 这个就是热部署的原理。
其他方面自定义类加载器没有什么用。
强引用 软引用(如果内部不够了,则回收) 场景:缓存框架