Java运行时环境之ClassLoader类加载机制详解

背景:听说ClassLoader类加载机制是进入BAT的必经之路。

ClassLoader总述:

普通的Java开发其实用到ClassLoader的地方并不多,但是理解透彻ClassLoader类的加载机制,无论是对我们编写更高效的代码还是进BAT都大有裨益;而从“黄埔军校”出来的我对ClassLoader的理解都是借鉴了很多书籍和博客,站在了各大博主的肩膀上,感谢你们!上菜,Classloader最主要的作用就是将Java字节码文件(后缀为.class)加载到JVM中,JVM在启动时不会一次性加载所有的class文件,而是根据需要动态加载class文件,毕竟一次性加载太多jar包的class文件JVM吃不消;下面主要研究Bootstrap ClassLoader、Extention ClassLoader和AppClassLoader这三种类加载器。

谈到ClassLoader就想到我们安装JDK的时候都会在控制台输入java、javac验证是否安装成功,而这个javac就是Java ClassLoader,测试是否能把Java源文件正确编译成Java字节码文件,下面的截图就是个javac的小例子,javac之后加载器把Java源文件编译成TestClassLoader.class字节码文件。

由于下面要讲到ClassLoader的加载路径,这里顺便把Java的环境变量也复习一遍。

JAVA_HOME:

指的是安装JDK的位置,如:JAVA_HOME="/Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home"

PATH:

配置PATH(程序的路径)的作用将就是能够在命令行窗口直接键入程序的名字了,而不再需要键入它的全路径,比如上面代码中我用的到javac和java两个命令。如:PATH=".$PATH:$JAVA_HOME/bin" ;就是在JAVA_HOME路径上添加了JDK下的bin目录即可。

CLASSPATH:

CLASSPATH就是指向jar包的路径,如:PATH=".$PATH:$JAVA_HOME/bin" ; "."表示当前目录。

ClassLoader类加载流程:

三个Class Loader的执行顺序是:Bootstrap CLassloder -> Extention ClassLoader -> AppClassLoader;

1、Bootstrap CLassloder是最顶层的加载类,主要是加载核心类库,也就是%JRE_HOME%\lib下的rt.jar、resources.jar、charsets.jar和class等资源;并且,可以通过启动JVM时指定-Xbootclasspath和路径来改变Bootstrap ClassLoader的加载目录,下面有个小荔子。

2、Extention ClassLoader是扩展的类加载器,其加载的是目录%JRE_HOME%\lib\ext目录下的jar包和class文件;它同样也可以加载-D java.ext.dirs选项指定的目录。

3、Appclass Loader是用于加载当前应用的classpath的所有类,其也称为SystemAppClass。

另外有兴趣的还可以看下Launcher类的源码,源码中规定了三个加载器的环境属性分别为B:sun.boot.class.path、E:java.ext.dirs和A:java.class.path;下面通过代码来简单测试写,如图:

打印输出结果:

BootstrapClassLoader:
        /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/resources.jar:
            /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/rt.jar:
                /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/sunrsasign.jar:
                    /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/jsse.jar:
                        /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/jce.jar:
                            /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/charsets.jar:
                                /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/jfr.jar:
                                    /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/classes

ExtClassLoader:
        /Users/apple/Library/Java/Extensions:
            /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/ext:
                /Library/Java/Extensions:/Network/Library/Java/Extensions:
                    /System/Library/Java/Extensions:/usr/lib/java

AppClassLoader:
        /TJT/Eclipse/workspace/tjt/bin:
            /Library/Java/JavaVirtualMachines/jdk1.8.0_191.jdk/Contents/Home/jre/lib/rt.jar

为了更好的理解三者之间加载的关系,我们来测试一个类的加载器和它的父类加载以及一些不是我们创建的类如String、Double、int等基础类:

从上图中可用看出,自己编写的类Test2.class文件是由AppClassLoader加载的,并且AppClassLoader有父加载器ExtClassLoader,但ExtClassLoader的父加载器为null;Double.class这个Java基础类的加载器为null,其父加载也为空且程序还会报空指针异常错误;其实呢,Double.class是有Bootstrap CLassLoader加载的,也并不是每个加载器都有父加载器;总的来说就是JVM启动时通过Bootstrap类加载器加载rt.jar等核心jar包中的class文件,诸如一些int.class,String.class都是由它加载;JVM初始化sun.misc.Launcher并创建Extension ClassLoader和AppClassLoader实例,且将ExtClassLoader设置为AppClassLoader的父加载器;而Bootstrap虽然没有父加载器,但是它却可以作为一个ClassLoader的父加载器;另外,一个ClassLoader创建时如果没有指定parent,那么它的parent默认就是AppClassLoader;

双亲委托:

当一个类加载器查找class和resource时,是通过“委托模式”进行的,它首先会判断这个class是不是已经加载成功,如果没有加载的话它并不是自己进行查找,而是先通过父加载器,然后递归下去,直到Bootstrap ClassLoader,如果Bootstrap classloader找到了,直接返回,如果没有找到,则一级一级返回,最后是由自身去查找这些对象;这种机制就叫做双亲委托。

从上图可用看出ClassLoader的加载序列,委托是从下往上,查找过程则是从上向下的,以下有几个注意事项:

1、一个AppClassLoader查找资源时,首先会查看缓存是否有,若有则从缓存中获取,否则委托给父加载器;

2.、重复第一步的递归操作,查询类是否已被加载;

3.、如果ExtClassLoader也没有加载过,则由Bootstrap ClassLoader加载,它首先也会查找缓存,如果没有找到的话,就去找自己的规定的路径下,也就是sun.mic.boot.class下面的路径,找到就返回,找不到就让子加载器自己去找。

4.、Bootstrap ClassLoader如果没有查找成功,则ExtClassLoader自己在java.ext.dirs路径中去查找,查找成功就返回,查找不成功则再向下让子加载器找。

5.、若是ExtClassLoader查找不成功,则由ppClassLoader在java.class.path路径下自己查找查找,找到就返回,如果没有找到就让子类找,如果没有子类则会抛出各种异常。

自定义CLassLoader:

在ClassLoader中有四个很重要实用的方法loadClass()、findLoadedClass()、findClass()、defineClass(),可以用来创建属于自己的类的加载方式;比如我们需要动态加载一些东西,或者从C盘某个特定的文件夹加载一个class文件,又或者从网络上下载class主内容然后再进行加载等。分三步搞定:

1、编写一个类继承ClassLoader抽象类;

2、重写findClass()方法;

3、在findClass()方法中调用defineClass()方法即可实现自定义ClassLoader;

需求:自定义一个classloader其默认加载路径为"/TJT/Code"下的jar包和资源。首先创建一个Test.java,然后javac编译并把生成的Test.class文件放到"/TJT/Code"路径下,然后再编写一个DiskClassLoader继承ClassLoader,最后通过FindClassLoader的测试类,调用再Test.class里面的一个find()方法。

package www.baidu;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

public class DiskClassLoader extends ClassLoader{
//自定义classLoader能将class二进制内容转换成Class对象
 private String myPath;

 public DiskClassLoader(String path) {
  myPath = path;
 }

 //findClass()方法中定义了查找class的方法
 @Override
 protected Class<?> findClass(String name) throws ClassNotFoundException{
  String fileName = getFileName(name);
  File file = new File(myPath,fileName);
  try {
   FileInputStream is = new FileInputStream(file);
   ByteArrayOutputStream bos = new ByteArrayOutputStream();
   int len = 0;
   try {
    while((len = is.read()) != -1) {
     bos.write(len);
    }
   } catch (IOException e) {
    e.printStackTrace();
   }
   byte[] data = bos.toByteArray();
   is.close();
   bos.close();
   //数据通过defineClass()生成了Class对象
   return defineClass(name, data,0,data.length );
  } catch (Exception e) {
   e.printStackTrace();
  }
  return super.findClass(name);
 }

 private String getFileName(String name) {
  int lastIndexOf = name.lastIndexOf('.');
  if (lastIndexOf == -1) {
   return name + ".class";
  }else {
   return name.substring(lastIndexOf + 1) + ".class";
  }
 }
}

测试结果如下:找到了自定义的加载路径并且调用了类中的find()方法

package www.baidu;
import java.lang.reflect.Method;

public class FindClassLoader {
 public static void main(String[] args) throws ClassNotFoundException {
  //创建自定义classloader对象
  DiskClassLoader diskL = new DiskClassLoader("/TJT/Code");
  System.out.println("classloader is: "+diskL);
  try {
    //加载class文件
   Class clazz = diskL.loadClass("www.baidu.Test");
   if (clazz != null) {
    Object object = clazz.newInstance();
    Method declaredMethod = clazz.getDeclaredMethod("find", null);
    //通过反射调用Test类的find()方法
    declaredMethod.invoke(object, null);
   }
  } catch (Exception e) {
   e.printStackTrace();
  }
 }
}

总结:

除此之外,ClassLoader还可以进行程序加密(比如你写了比较骚的jar包),这样我们就可以在程序中加载特定了类,并且这个类只能被我们自定义的加载器进行加载,提高了程序的安全性,但是用的不多;反正我们在项目上是不允许用ClassLoader加密,宁愿裸奔,了解一下。另外就是tomcat的类加载机制也是遵循双亲委派机制的,并且大部分的加载机制和JVM类加载机制一样,理解了Bootstrap ClassLoader、Extention ClassLoader和AppClassLoader这三种加载器后再看tomcat类的加载就可以横着走了。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Android开发中类加载器DexClassLoader的简单使用讲解

    简介 "类装载器"(ClassLoader),顾名思义,就是用来动态装载class文件的.标准的Java SDK中有个ClassLoader类,借助此类可以装载需要的class文件,前提是ClassLoader类初始化必须制定class文件的路径. import关键字引用的类文件和ClassLoader动态加载类的区别: import引用类的两个特点: 1.必须存在于本地,当程序运行该类时,内部类装载器会自动装载该类. 2.编译时必须在现场,否则编译过程会因找不到引用文件而不能正常编译

  • classloader类加载器_基于java类的加载方式详解

    基础概念 Classloader 类加载器,用来加载 Java 类到 Java 虚拟机中.与普通程序不同的是.Java程序(class文件)并不是本地的可执行程序.当运行Java程序时,首先运行JVM(Java虚拟机),然后再把Java class加载到JVM里头运行,负责加载Java class的这部分就叫做Class Loader. JVM本身包含了一个ClassLoader称为Bootstrap ClassLoader,和JVM一样,BootstrapClassLoader是用本地代码实现

  • 详解Android类加载ClassLoader

    基本知识 Java的类加载设计了一套双亲代理的模式,使得用户没法替换系统的核心类,从而让应用更安全.所谓双亲代理就是指,当加载类的时候首先去Bootstrap中加载类,如果没有则去Extension中加载,如果再没有才去AppClassLoader中去加载.从而实现安全和稳定. Java ClassLoader BootstrapClassLoader 引导类加载器 ,用来加载Java的核心库.通过底层代码来实现的,基本上只要parent为null,那就表示引导类加载器. 比如:charsets

  • ClassLoader类加载源码解析

    Java类加载器 1.BootClassLoader: 用于加载Android Framework层class文件. 2.PathClassLoader: 用于Android应用程序类加载器.可以加载指定的dex,jar.zip.zpk中的classes.dex 3.DexClassLoader:加载指定的dex,以及jar.zip.apk中的classes.dex 源码解析 1.ClassLoader中提供loadClass用于加载指定类 //ClassLoader.java public C

  • Java中ClassLoader类加载学习总结

    双亲委派模型 类加载这个概念应该算是Java语言的一种创新,目的是为了将类的加载过程与虚拟机解耦,达到"通过类的全限定名来获取描述此类的二进制字节流"的目的.实现这个功能的代码模块就是类加载器.类加载器的基本模型就是大名鼎鼎的双亲委派模型(Parents Delegation Model).听上去很牛掰,其实逻辑很简单,在需要加载一个类的时候,我们首先判断该类是否已被加载,如果没有就判断是否已被父加载器加载,如果还没有再调用自己的findClass方法尝试加载.基本的模型就是这样(盗图

  • Java运行时环境之ClassLoader类加载机制详解

    背景:听说ClassLoader类加载机制是进入BAT的必经之路. ClassLoader总述: 普通的Java开发其实用到ClassLoader的地方并不多,但是理解透彻ClassLoader类的加载机制,无论是对我们编写更高效的代码还是进BAT都大有裨益:而从"黄埔军校"出来的我对ClassLoader的理解都是借鉴了很多书籍和博客,站在了各大博主的肩膀上,感谢你们!上菜,Classloader最主要的作用就是将Java字节码文件(后缀为.class)加载到JVM中,JVM在启动时

  • Java运行时动态生成类实现过程详解

    最近一个项目中利用规则引擎,提供用户拖拽式的灵活定义规则.这就要求根据数据库数据动态生成对象处理特定规则的逻辑.如果手写不仅每次都要修改代码,还要每次测试发版,而且无法灵活根据用户定义的规则动态处理逻辑.所以想到将公共逻辑写到父类实现,将特定逻辑根据字符串动态生成子类处理.这就可以一劳永逸解决这个问题. 那就着手从Java如何根据字符串模板在运行时动态生成对象. Java是一门静态语言,通常,我们需要的class在编译的时候就已经生成了,为什么有时候我们还想在运行时动态生成class呢? 经过一

  • jvm虚拟机类加载机制详解

    目录 1 概述 2 类的加载时机 3 类的加载过程 3.1 加载 3.2 验证 3.3 准备 3.4 解析 3.5 初始化 4 类加载器 4.1 双亲委派模型 4.2 破坏双亲委派模型 1 概述 ​ Java虚拟机把描述类的数据从Class文件加载到内存, 并对数据进行校验.转化解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这个过程称为虚拟机的类加载机制.在Java语言中,类型的加载.连接和初始化都是在程序运行期间完成的. 2 类的加载时机 ​ 一个类型从被加载到虚拟机内存中开始,到

  • JVM分析之类加载机制详解

    目录 1.前言 2.类加载是什么 3.类加载过程 3.1 加载 3.2 链接 3.3 初始化 4.总结 1.前言 JVM内部架构包含类加载器.内存区域.执行引擎等.日常开发中,我们编写的java文件被编译成class文件后,jvm会进行加载并运行使用类.本次仅对JVM加载部分进行分析,了解并掌握加载机制. 2.类加载是什么 类加载是一种过程,是将class文件加载到jvm内存的过程.当代码逻辑中需要引用类时,通过类加载器加载引用类对象并存放堆中,以供代码调用. 3.类加载过程 注:类加载过程包含

  • Java并发中的Fork/Join 框架机制详解

    什么是 Fork/Join 框架 Fork/Join 框架是一种在 JDk 7 引入的线程池,用于并行执行把一个大任务拆成多个小任务并行执行,最终汇总每个小任务结果得到大任务结果的特殊任务.通过其命名也很容易看出框架主要分为 Fork 和 Join 两个阶段,第一阶段 Fork 是把一个大任务拆分为多个子任务并行的执行,第二阶段 Join 是合并这些子任务的所有执行结果,最后得到大任务的结果. 这里不难发现其执行主要流程:首先判断一个任务是否足够小,如果任务足够小,则直接计算,否则,就拆分成几个

  • java开发分布式服务框架Dubbo原理机制详解

    目录 前言 Dubbo框架有以下部件 Consumer Provider Registry Monitor Container 架构 高可用性 框架设计 服务暴露过程 服务消费过程 前言 在介绍Dubbo之前先了解一下基本概念: Dubbo是一个RPC框架,RPC,即Remote Procedure Call(远程过程调用),相对的就是本地过程调用,在分布式架构之前的单体应用架构和垂直应用架构运用的都是本地过程调用.它允许程序调用另外一个地址空间(通常是网络共享的另外一台机器)的过程或函数,并且

  • Java在运行时识别类型信息的方法详解

    前言 在日常的学习工作当中,有一些知识是我们在读书的时候就能够习得:但有一些知识不是的,需要在实践的时候才能得到真知--这或许就是王阳明提倡的"知行合一". 在Java中,并不是所有的类型信息都能在编译阶段明确,有一些类型信息需要在运行时才能确定,这种机制被称为RTTI,英文全称为Run-Time Type Identification,即运行时类型识别,有没有一点"知行合一"的味道?运行时类型识别主要由Class类实现. 01 Class类 在Java中,我们常用

  • 深入理解JVM之类加载机制详解

    本文实例讲述了深入理解JVM之类加载机制.分享给大家供大家参考,具体如下: 概述 虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 与那些在编译时需要进行链接工作的语言不同,在Java语言里,类型的加载.连接和初始化过程都是在程序运行期间完成的,例如import java.util.*下面包含很多类,但是,在程序运行的时候,虚拟机只会加载哪些我们程序需要的类.这种策略虽然会令类加载时稍微增加

  • 面试必时必问的JVM 类加载机制详解

    目录 前言 正文 1.类加载的过程. 1)加载 2)验证 3)准备 4)解析 5)初始化 2.Java 虚拟机中有哪些类加载器? 1)启动类加载器(Bootstrap ClassLoader): 2)扩展类加载器(Extension ClassLoader): 3)应用程序类加载器(Application ClassLoader): 3.什么是双亲委派模型? 4.为什么使用双亲委派模式? 5.有哪些场景破坏了双亲委派模型? 6.为什么要破坏双亲委派模型? 7.如何破坏双亲委派模型? 8.Tomc

  • JVM类加载机制详解

    一.先看看编写出的代码的执行过程: 二.研究类加载机制的意义 从上图可以看出,类加载是Java程序运行的第一步,研究类的加载有助于了解JVM执行过程,并指导开发者采取更有效的措施配合程序执行. 研究类加载机制的第二个目的是让程序能动态的控制类加载,比如热部署等,提高程序的灵活性和适应性. 三.类加载的一般过程 原理:双亲委托模式 1.寻找jre目录,寻找jvm.dll,并初始化JVM: 2.产生一个Bootstrap Loader(启动类加载器): 3.Bootstrap Loader自动加载E

随机推荐