JVM 虚拟器

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对象,用来封装类在方法区内的数据结构

JVM 虚拟器_第1张图片


•类的加载的最终产品是位于堆区中的Class对象
•Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口


二, 自定义类加载器
用用户自定义的类加载是可以被卸载的。 这个就是热部署的原理。
其他方面自定义类加载器没有什么用。


强引用 软引用(如果内部不够了,则回收)  场景:缓存框架

你可能感兴趣的:(jvm)