黑马程序员技术交流社区

标题: 【上海校区】JVM系列(五) - JVM类加载机制详解 [打印本页]

作者: 不二晨    时间: 2018-7-24 13:36
标题: 【上海校区】JVM系列(五) - JVM类加载机制详解
本文将由浅及深,介绍Java类加载的过程和原理,进一步对类加载器的进行源码分析,完成一个自定义的类加载器。


正文(一). 类加载器是什么类加载器简言之,就是用于把.class文件中的字节码信息转化为具体的java.lang.Class对象的过程的工具。
具体过程:
Class对象的创建过程描述:

(二). 类加载的过程类加载的过程分为三个步骤(五个阶段) :加载 -> 连接验证准备解析)-> 初始化
加载验证准备初始化这四个阶段发生的顺序是确定的,而解析阶段可以在初始化阶段之后发生,也称为动态绑定晚期绑定
类加载的过程描述:

1. 加载加载:查找并加载类的二进制数据的过程。
加载的过程描述:2. 连接连接:包括验证准备解析三步。
a). 验证验证:确保被加载的类的正确性。验证是连接阶段的第一步,用于确保Class字节流中的信息是否符合虚拟机的要求。
具体验证形式:b). 准备准备:为类的静态变量分配内存,并将其初始化为默认值。准备过程通常分配一个结构用来存储类信息,这个结构中包含了类中定义的成员变量方法接口信息等。
具体行为:c). 解析解析:把类中对常量池内的符号引用转换为直接引用
解析动作主要针对类或接口字段类方法接口方法方法类型方法句柄调用点限定符等7类符号引用进行。
3. 初始化初始化:对类静态变量赋予正确的初始值 (注意和连接时的解析过程区分开)。
初始化的目标初始化的步骤初始化的时机(三). 类的主动引用和被动引用主动引用主动引用:在类加载阶段,只执行加载连接操作,不执行初始化操作。
主动引用的几种形式主动引用1 - main方法在初始类中代码示例:
public class OptimisticReference0 {    static {        System.out.println(OptimisticReference0.class.getSimpleName() + " is referred!");    }    public static void main(String[] args) {        System.out.println();    }}复制代码运行结果:
OptimisticReference0 is referred!
主动引用2 – 创建子类会触发父类的初始化代码示例:
public class OptimisticReference1 {    public static class Parent {        static {            System.out.println(Parent.class.getSimpleName() + " is referred!");        }    }    public static class Child extends Parent {        static {            System.out.println(Child.class.getSimpleName() + " is referred!");        }    }    public static void main(String[] args) {        new Child();    }}复制代码运行结果:
Parent is referred!Child is referred!
主动引用3 – 访问一个类静态变量代码示例:
public class OptimisticReference2 {    public static class Child {        protected static String name;        static {            System.out.println(Child.class.getSimpleName() + " is referred!");            name = "Child";        }    }    public static void main(String[] args) {        System.out.println(Child.name);    }}复制代码运行结果:
Child is referred!Child
主动引用4 – 对类的静态变量进行赋值代码示例:
public class OptimisticReference3 {    public static class Child {        protected static String name;        static {            System.out.println(Child.class.getSimpleName() + " is referred!");        }    }    public static void main(String[] args) {        Child.name = "Child";    }}复制代码运行结果:
Child is referred!
主动引用5 – 使用java.lang.reflect包提供的反射机制代码示例:
public class OptimisticReference4 {    public static void main(String[] args) throws ClassNotFoundException {        Class.forName("org.ostenant.jdk8.learning.examples.reference.optimistic.Child");    }}复制代码运行结果:
Child is referred!
被动引用被动引用: 在类加载阶段,会执行加载连接初始化操作。
被动引用的几种形式:
被动引用1 – 子类引用父类的的静态字段,不会导致子类初始化代码示例:
public class NegativeReference0 {    public static class Parent {        public static String name = "Parent";        static {            System.out.println(Parent.class.getSimpleName() + " is referred!");        }    }    public static class Child extends Parent {        static {            System.out.println(Child.class.getSimpleName() + " is referred!");        }    }    public static void main(String[] args) {        System.out.println(Child.name);    }}复制代码运行结果:
Parent is referred!Parent
被动引用2 – 定义类的数组引用而不赋值,不会触发此类的初始化代码示例:
public class NegativeReference1 {    public static class Child {        static {            System.out.println(Child.class.getSimpleName() + " is referred!");        }    }    public static void main(String[] args) {        Child[] childs = new Child[10];    }}复制代码运行结果:
无输出
被动引用3 – 访问类定义的常量,不会触发此类的初始化示例代码:
public class NegativeReference2 {    public static class Child {        public static final String name = "Child";        static {            System.out.println(Child.class.getSimpleName() + " is referred!");        }    }    public static void main(String[] args) {        System.out.println(Child.name);    }}复制代码运行结果:
Child
(四). 三种类加载器类加载器:类加载器负责加载程序中的类型(类和接口),并赋予唯一的名字予以标识。
类加载器的组织结构

类加载器的关系类加载器的作用
Class Loader
实现方式具体实现类负责加载的目标
Bootstrap Loader
C++由C++实现%JAVA_HOME%/jre/lib/rt.jar以及-Xbootclasspath参数指定的路径以及中的类库
Extension ClassLoader
Javasun.misc.Launcher$ExtClassLoader%JAVA_HOME%/jre/lib/ext路径下以及java.ext.dirs系统变量指定的路径中类库
Application ClassLoader
Javasun.misc.Launcher$AppClassLoaderClasspath以及-classpath、-cp指定目录所指定的位置的类或者是jar文档,它也是Java程序默认的类加载器类加载器的特点类加载器的隔离问题每个类装载器都有一个自己的命名空间用来保存已装载的类。当一个类装载器装载一个类时,它会通过保存在命名空间里的类全局限定名(Fully Qualified Class Name) 进行搜索来检测这个类是否已经被加载了。
JVM 及 Dalvik 对类唯一的识别是 ClassLoader id + PackageName + ClassName,所以一个运行程序中是有可能存在两个包名类名完全一致的类的。并且如果这两个不是由一个 ClassLoader 加载,是无法将一个类的实例强转为另外一个类的,这就是 ClassLoader 隔离性。
为了解决类加载器的隔离问题,JVM引入了双亲委托机制
(五). 双亲委托机制核心思想:其一,自底向上检查类是否已加载;其二,自顶向下尝试加载类
具体加载过程源码分析ClassLoader.class
ClassLoader通过loadClass()方法实现了双亲委托机制,用于类的动态加载
loadClass()本身是一个递归向上调用的过程。
查找当前类加载器的缓存中是否已经加载目标类。findLoadedClass()实际调用了底层的native方法findLoadedClass0()。
查找最顶端Bootstrap类加载器的是否已经加载目标类。同样,findBootstrapClassOrNull()实际调用了底层的native方法findBootstrapClass()。
ClassLoader是java.lang包下的抽象类,也是所有类加载器(除了Bootstrap)的基类,findClass()是ClassLoader对子类提供的加载目标类的抽象方法。
注意:Bootstrap ClassLoader并不属于JVM的层次,它不遵守ClassLoader的加载规则,Bootstrap classLoader并没有子类。
defineClass()是ClassLoader向子类提供的方法,它可以将.class文件的二进制数据转换为合法的java.lang.Class对象。
(六). 类的动态加载类的几种加载方式Class.forName()和ClassLoader.loadClass()(七). 对象的初始化对象的初始化顺序静态变量/静态代码块 -> 普通代码块 -> 构造函数
对象的初始化示例Parent.java

Children.java

Tester.java

测试结果:

测试结果表明:JVM在创建对象时,遵守以上对象的初始化顺序。
(八). 自定义类加载器编写自己的类加载器在源码分析阶段,我们已经解读了如何实现自定义类加载器,现在我们开始自己的类加载器。
Step 1:定义待加载的目标类Parent.java和Children.java。
Parent.java
package org.ostenant.jdk8.learning.examples.classloader.custom;public class Parent {    protected static String CLASS_NAME;    protected static String CLASS_LOADER_NAME;    protected String instanceID;        // 1.先执行静态变量和静态代码块(只在类加载期间执行一次)    static {        CLASS_NAME = Parent.class.getName();        CLASS_LOADER_NAME = Parent.class.getClassLoader().toString();        System.out.println("Step a: " + CLASS_NAME + " is loaded by " + CLASS_LOADER_NAME);    }    // 2.然后执行变量和普通代码块(每次创建实例都会执行)    {        instanceID = this.toString();        System.out.println("Step c: Parent instance is created: " + CLASS_LOADER_NAME + " -> " + instanceID);    }    // 3.然后执行构造方法    public Parent() {        System.out.println("Step d: Parent instance:" + instanceID + ", constructor is invoked");    }    public void say() {        System.out.println("My first class loader...");    }}复制代码Children.java
package org.ostenant.jdk8.learning.examples.classloader.custom;public class Children extends Parent {    static {        CLASS_NAME = Children.class.getName();        CLASS_LOADER_NAME = Children.class.getClassLoader().toString();        System.out.println("Step b: " + CLASS_NAME + " is loaded by " + CLASS_LOADER_NAME);    }    {        instanceID = this.toString();        System.out.println("Step e: Children instance is created: " + CLASS_LOADER_NAME + " -> " + instanceID);    }    public Children() {        System.out.println("Step f: Children instance:" + instanceID + ", constructor is invoked");    }    public void say() {        System.out.println("My first class loader...");    }}复制代码
Step 2:实现自定义类加载器CustomClassLoader
CustomClassLoader.java
public class CustomClassLoader extends ClassLoader {    private String classPath;    public CustomClassLoader(String classPath) {        this.classPath = classPath;    }    @Override    protected Class<?> findClass(String name) throws ClassNotFoundException {        Class<?> c = findLoadedClass(name); // 可省略        if (c == null) {            byte[] data = loadClassData(name);            if (data == null) {                throw new ClassNotFoundException();            }            return defineClass(name, data, 0, data.length);        }        return null;    }    protected byte[] loadClassData(String name) {        try {            // package -> file folder            name = name.replace(".", "//");            FileInputStream fis = new FileInputStream(new File(classPath + "//" + name + ".class"));            ByteArrayOutputStream baos = new ByteArrayOutputStream();            int len = -1;            byte[] b = new byte[2048];            while ((len = fis.read(b)) != -1) {                baos.write(b, 0, len);            }            fis.close();            return baos.toByteArray();        } catch (IOException e) {            e.printStackTrace();        }        return null;    }}复制代码
Step 3:测试类加载器的加载过程
CustomerClassLoaderTester.java
    private static final String CHILDREN_SOURCE_CODE_NAME = SOURCE_CODE_LOCATION + "Children.java";    private static final String PARENT_SOURCE_CODE_NAME = SOURCE_CODE_LOCATION + "Parent.java";    private static final List<String> SOURCE_CODE = Arrays.asList(CHILDREN_SOURCE_CODE_NAME, PARENT_SOURCE_CODE_NAME);    static {        SOURCE_CODE.stream().map(path -> new File(path))            // 路径转文件对象            .filter(f -> !f.isDirectory())            // 文件遍历            .forEach(f -> {            // 拷贝后源代码            File targetFile = copySourceFile(f);            // 编译源代码            compileSourceFile(targetFile);        });    }复制代码    protected static File copySourceFile(File f) {        BufferedReader reader = null;        BufferedWriter writer = null;        try {            reader = new BufferedReader(new FileReader(f));            // package ...;            String firstLine = reader.readLine();            StringTokenizer tokenizer = new StringTokenizer(firstLine, " ");            String packageName = "";            while (tokenizer.hasMoreElements()) {                String e = tokenizer.nextToken();                if (e.contains("package")) {                    continue;                } else {                    packageName = e.trim().substring(0, e.trim().length() - 1);                }            }            // package -> path            String packagePath = packageName.replace(".", "//");            // java file path            String targetFileLocation = TARGET_CODE_LOCALTION + "//" + packagePath + "//";            String sourceFilePath = f.getPath();            String fileName = sourceFilePath.substring(sourceFilePath.lastIndexOf("\\") + 1);            File targetFile = new File(targetFileLocation, fileName);            File targetFileLocationDir = new File(targetFileLocation);            if (!targetFileLocationDir.exists()) {                targetFileLocationDir.mkdirs();            }            // writer            writer = new BufferedWriter(new FileWriter(targetFile));            // 写入第一行            writer.write(firstLine);            writer.newLine();            writer.newLine();            String input = "";            while ((input = reader.readLine()) != null) {            writer.write(input);                writer.newLine();            }            return targetFile;        } catch (FileNotFoundException e) {            e.printStackTrace();        } catch (IOException e) {            e.printStackTrace();        } finally {            try {                reader.close();                writer.close();            } catch (IOException e) {                e.printStackTrace();            }        }        return null;    }复制代码    protected static void compileSourceFile(File f) {        try {            JavaCompiler javaCompiler = ToolProvider.getSystemJavaCompiler();            StandardJavaFileManager standardFileManager = javaCompiler.getStandardFileManager(null, null, null);            Iterable<? extends JavaFileObject> javaFileObjects = standardFileManager.getJavaFileObjects(f);            // 执行编译任务            CompilationTask task = javaCompiler.getTask(null, standardFileManager, null, null, null, javaFileObjects);            task.call();            standardFileManager.close();        } catch (Exception e) {            e.printStackTrace();        }    }复制代码    @Test    public void test() throws Exception {        // 创建自定义类加载器        CustomClassLoader classLoader = new CustomClassLoader(TARGET_CODE_LOCALTION); // E://myclassloader//classpath        // 动态加载class文件到内存中(无连接)        Class<?> c = classLoader.loadClass("org.ostenant.jdk8.learning.examples.classloader.custom.Children");        // 通过反射拿到所有的方法        Method[] declaredMethods = c.getDeclaredMethods();        for (Method method : declaredMethods) {            if ("say".equals(method.getName())) {                // 通过反射拿到children对象                Object children = c.newInstance();                // 调用children的say()方法                method.invoke(children);                break;            }        }    }复制代码测试编写的类加载器(一). 测试场景一测试结果输出:

测试结果分析:
我们成功创建了Children对象,并通过反射调用了它的say()方法。然而查看控制台日志,可以发现类加载使用的仍然是AppClassLoader,CustomClassLoader并没有生效。
查看CustomClassLoader的类加载目录:

类目录下有我们拷贝编译的Parent和Chidren文件。
分析原因:
由于项目空间中的Parent.java和Children.java,在拷贝后并没有移除。导致AppClassLoader优先在其Classpath下面找到并成功加载了目标类。
(二). 测试场景二测试结果输出:
测试结果分析:
我们成功通过自定义类加载器加载了目标类。创建了Children对象,并通过反射调用了它的say()方法。
至此,我们自己的一个简单的类加载器就完成了!


作者:零壹技术栈
链接:https://juejin.im/post/5b4deaa3e51d4519596b7f27




作者: 不二晨    时间: 2018-7-24 14:01
奈斯
作者: 摩西摩西OvO    时间: 2018-7-26 09:12

作者: 吴琼老师    时间: 2018-7-26 15:31





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2