类加载机制:
JVM把class文件加载到内存,并对数据进行校验、解析、和初始化,最终形成JVM可以直接使用的Java类型的过程。
加载:
把class文件字节码内容加载到内存中,并将这些静态数据转换成方法区中的数据结构,在堆中生成一个代表这个类的java.lang.Class对象,作为这个方法区的访问入口。
链接:将二进制代码合并到JVM的运行状态之中的过程。
—验证:确保加载的类的信息符合JVM规范,没有安全方面的问题。
—准备:正式为类变量(static变量)分配内存并设置类变量初始值的阶段,这些内存将在方法区中进行分配。
—解析:虚拟机常量池的符号引用替换为直接应用。
初始化:
初始化阶段是执行类构造器<clinit>()方法过程,类构造器<clinit>()方法是由编译器自动收集类中所有类变量的赋值动作和静态语句块(static块)中语句合成的过程。
当初始化一个类的时候,如果发现其父类没有被初始化,则先初始化其父类。
虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确的加锁和同步。
当访问一个Java类的静态域时,只有真正声明这个域的类才会被初始化。
public class classLoader {
static {
System.out.println("静态初始化classLoader");
}
public static void main(String[] args)
{
System.out.println("main方法加载");
A a = new A();
System.out.println(A.width);
A a2 = new A(); //这里代表类加载初始化只有一次,然后调用的是对象的方法。
}
}
class A extends B{
public static int width=100;
static {
System.out.println("静态初始化A"); //这里静态方法静态变量会合并到一起
width = 300; //也就是width=300;
}
public A()
{
System.out.println("创建A的对象");
}
}
class B {
static {
System.out.println("静态初始化B");
}
}
类加载器的层次结构(树状结构):
• 引导类加载器(bootstrap class loader) – 它用来加载 Java 的核心库(JAVA_HOME/jre/lib/rt.jar,或sun.boot.class.path路径下的 内容),是用原生代码来实现的,并不继承自 java.lang.ClassLoader。 – 加载扩展类和应用程序类加载器。并指定他们的父类加载器。
• 扩展类加载器(extensions class loader) – 用来加载 Java 的扩展库(JAVA_HOME/jre/ext/*.jar,或java.ext.dirs路径下的内容) 。 Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。 – 由sun.misc.Launcher$ExtClassLoader实现
• 应用程序类加载器(application class loader) – 它根据 Java 应用的类路径(classpath, java.class.path 路径下的内容)来加载 Java 类。 一般来说,Java 应用的类都是由它来完成加载的。 – 由sun.misc.Launcher$AppClassLoader实现
• 自定义类加载器 – 开发人员可以通过继承 java.lang.ClassLoader类的方式 实现自己的类加载器,以满足一些特殊的需求。
类加载器的代理模式:
• 代理模式 – 交给其他加载器来加载指定的类
• 双亲委托机制 – 就是某个特定的类加载器在接到加载类的请求时,首先将加载任务委 托给父类加载器,依次追溯,直到最高的爷爷辈的,如果父类加载器 可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载 任务时,才自己去加载。 – 双亲委托机制是为了保证 Java 核心库的类型安全。
• 这种机制就保证不会出现用户自己能定义java.lang.Object类的情况。 – 类加载器除了用于加载类,也是安全的最基本的屏障。
• 双亲委托机制是代理模式的一种 – 并不是所有的类加载器都采用双亲委托机制。 – tomcat服务器类加载器也使用代理模式,所不同的是它是首先尝试去加载某个类,如果找不到再代理给父类加载器。 这与一般类加载器的顺序是相反的