二十三 类加载机制

2018-11-28  本文已影响0人  BeYearn
  1. 类的加载过程
  1. 加载(loading) 它是将字节码数据从不同的数据源读取到jvm中, 并映射为jvm认可的数据结构(Class对象). 这里的数据源可以是各种各样的形态,如jar文件,class文件甚至是网络数据源; 如果输入数据不是ClassFile的结构, 则会抛出ClassFormatError. 这个阶段是用户参与阶段, 我们可以自定义类加载器, 去实现自己的类加载过程
  2. 第二阶段 链接(Linking) 这是核心步骤, 即将原始的类定义信息平滑转入jvm运行的过程中.
    a. 验证(Vertification) 虚拟机安全的重要保障, 需要核验字节信息是符合java虚拟机规范的,否则就VerfityError. 这样可防止恶意信息或不合规信息危害jvm运行.验证阶段有可能触发更多 class 的加载
    b. 准备(Preparation) 创建类或接口中的静态变量, 并初始化静态变量的初始值(区别于下面显示初始化阶段, 侧重点在于分配所需要的内存空间)
    c. 解析(Resolution) 在这一步会将常量池中的符号引用(symbolic reference)替换为直接引用。在[Java 虚拟机规范]中,详细介绍了类、接口、方法和字段等各个方面的解析。
  3. 初始化阶段(Initialization) 这一步真正去执行类初始化的代码逻辑, 包括静态字段赋值, 执行类定义中的静态初始块内的逻辑(编译器在编译阶段就会把这部分逻辑整理好). 父类型的初始化逻辑优先于当前类型的逻辑.
  1. parent-delegation model 双亲委派模型 (哈哈哈 啃老模型)
    如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把请求委托给父加载器去完成,依次向上,因此,所有的类加载请求最终都应该被传递到顶层的启动类加载器中,只有当父加载器在它的搜索范围中没有找到所需的类时,即无法完成该加载,子加载器才会尝试自己去加载该类。(避免重复加载java类型)

  2. 三种内建的类加载器(java8以前)

java -Djava.ext.dirs=your_ext_dir HelloWorld
java -Djava.system.class.loader=com.yourcorp.YourClassLoader HelloWorld

如果我们指定了这个参数,JDK 内建的应用类加载器就会成为定制加载器的父亲,这种方式通常用在类似需要改变双亲委派模式的场景。

关系如下图:


图片.png
  1. 字节码是平台无关的抽象, 而不是机器码, 所以java需要加载和解释/编译, 这些都导致java启动变慢. 一些通用的办法, 降低类加载的开销:
Java -Xshare:dump -XX:+UseAppCDS -XX:SharedArchiveFile=<jsa>  \
         -XX:SharedClassListFile=<classlist> -XX:SharedArchiveConfigFile=<config_file>

然后在应用程序启动时,指定归档文件,并开启 AppCDS。

Java -Xshare:on -XX:+UseAppCDS -XX:SharedArchiveFile=<jsa> yourApp

(如果恰好大量使用了运行时动态类加载,它的帮助就有限了)

  1. java hell问题
    当一个类或一个资源文件存在多个jar中,就会出现jar hell问题。
    可以通过以下代码来诊断方案:
try {
      ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
      String resourceName = "net/sf/cglib/proxy/MethodInterceptor.class";
        Enumeration<URL> urls = classLoader.getResources(resourceName);
  while(urls.hasMoreElements()){
        System.out.println(urls.nextElement());
            }
        } catch (IOException e) {
e.printStackTrace();
        }

//输出结果:
//jar:file:/D:/workspace/Test/lib/cglib-3.2.4.jar!/net/sf/cglib/proxy/MethodInterceptor.class
上一篇 下一篇

猜你喜欢

热点阅读