前言

JVM设计团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外面去实现,以便让应用程序自己决定如何去获取所需要的类。实现这个动作的代码模块称为“类加载器”。

类与类加载器

类加载器虽然只用户实现类的加载动作,但它在Java程序中起到的作用却远远不限于类加载阶段。每个类都有一个独立的类名称空间,在比较两个类是否“相等”,只有两个类是由同一个类加载器加载的前提下才有意义,否则即使两个类来源于同一个Class文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等。

这里的相等,包含Class对象的equals()方法、isAssignableFrom()方法、isInstance()方法的返回结果,也包括使用instanceof关键字做对象所属关系判定等情况。例如如下代码:

  1. public class ClassLoaderTest {
  2.  
  3. public static void main(String[] args) throws Exception{
  4.  
  5. ClassLoader myClassLoader = new ClassLoader() {
  6. /**
  7. * Loads the class with the specified <a href="#name">binary name</a>.
  8. * This method searches for classes in the same manner as the {@link
  9. * #loadClass(String, boolean)} method. It is invoked by the Java virtual
  10. * machine to resolve class references. Invoking this method is equivalent
  11. * to invoking {@link #loadClass(String, boolean) <tt>loadClass(name,
  12. * false)</tt>}.
  13. *
  14. * @param name The <a href="#name">binary name</a> of the class
  15. * @return The resulting <tt>Class</tt> object
  16. * @throws ClassNotFoundException If the class was not found
  17. */
  18. @Override
  19. public Class<?> loadClass(String name) throws ClassNotFoundException {
  20. try{
  21. String fileName = name.substring(name.lastIndexOf(".")+1)+".class";
  22.  
  23. InputStream inputStream = getClass().getResourceAsStream(fileName);
  24. if(null == inputStream){
  25. return super.loadClass(name);
  26. }
  27. byte[] b = new byte[inputStream.available()];
  28. inputStream.read(b);
  29. return defineClass(name,b,0,b.length);
  30. }catch (IOException e){
  31. throw new ClassNotFoundException(name);
  32. }
  33.  
  34. }
  35. };
  36.  
  37. Object obj = myClassLoader.loadClass("com.eurekaclient2.client2.shejimoshi.JVM.ClassLoaderTest").newInstance();
  38.  
  39. System.out.println("来源:"+obj.getClass());
  40.  
  41. System.out.println(obj instanceof com.eurekaclient2.client2.shejimoshi.JVM.ClassLoaderTest);
  42. }
  43.  
  44. }

运行结果:

  1. 来源:class com.eurekaclient2.client2.shejimoshi.JVM.ClassLoaderTest
  2. false

从运行结果中我们可以看出来,obj对象确实属于ClassLoaderTest类的对象,但是从运行结果的第二行中可以看出来,这个对象与ClassLoaderTest类做所属类型检查时返回的false,因为虚拟机中存在了两个ClassLoaderTest类,一个是由系统应用程序类加载器加载的,另一个是由我们自定义的类加载器加载的,虽然都来自同一个Class文件,但依然是两个独立的类。

双亲委派模型

从虚拟机的角度来讲,只存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader);另一种就是所有其他的类加载器,这些类加载器都是由Java语言实现,并且独立于虚拟机外部,并都继承自抽象类java.lang.ClassLoader。

从Java开人员的角度来看,类加载器可以分的更细一些,但是绝大部分java程序都会用到下面的这3种系统提供的类加载器。

启动类加载器(Bootstrap ClassLoader):它负责将放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中,并且是虚拟机识别的类库加载到虚拟机中。

扩展类加载器(Extension ClassLoader):它负责加载<JAVA_HOME>\lib\ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。

应用程序类加载器(Application ClassLoader):它一般被称为系统类加载器,负责加载用户类路径上所指定的类库,开发者可以直接使用这个类加载器,若应用程序中没有自定义过类加载器,一般情况下默认的就是这个应用程序类加载器。

我们的应用程序都是由这3种类加载器相互配合进行加载的,如果有需要,还可以加入自定义的类加载器。这些类加载器的关系如下图:

类加载器的之间的这种层次关系,称为类加载器的双亲委派模式(Parents Delegation Model)。这种模型要求,除了顶层外,其余的类加载器都应当有自己的的父类加载器。这里的子父关系一般不会以继承方式来实现,而是使用组合关系来复用父加载器的代码。

双亲委派模型的工作过程:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是这样,最终都应该传送到顶层的启动类加载器中,只有当父类反馈自己无法完成这个加载请求时,子加载器才会尝试自己去加载。

使用双亲委派模型来组织类加载器之间的关系,有一个显而易见测好处就是Java类随着它的类加载器一起具备了一种带有优先级的层次关系。

记得以前看到过一个面试题,题目大概意思是:能不能自己写一个类叫java.lang.String?

答案:不可以。原因就是因为JVM的类加载器采用的这种双亲委派模型,当我们写了一个类叫java.lang.String时,类加载器发现已经加载过一个同样的类了,不用加载了,直接使用就可以了。所以自己写的这个java.lang.String这个类可以编译通过,但是无法被加载运行。

实现双亲委派的代码集中在java.lang.ClassCloader的loadClass()方法中,逻辑很简单:首先检查自己是否已经被加载过,如果没有加载则调用父加载器的loadClass()方法,若父加载器为空则默认使用启动类加载器作为父加载器。如果父加载器加载失败,则抛出ClassNotFoundException异常后,再调用自己的findClass()方法进行加载。

loadClass的源码:

  1. /**
  2. * Loads the class with the specified <a href="#name">binary name</a>. The
  3. * default implementation of this method searches for classes in the
  4. * following order:
  5. *
  6. * <ol>
  7. *
  8. * <li><p> Invoke {@link #findLoadedClass(String)} to check if the class
  9. * has already been loaded. </p></li>
  10. *
  11. * <li><p> Invoke the {@link #loadClass(String) <tt>loadClass</tt>} method
  12. * on the parent class loader. If the parent is <tt>null</tt> the class
  13. * loader built-in to the virtual machine is used, instead. </p></li>
  14. *
  15. * <li><p> Invoke the {@link #findClass(String)} method to find the
  16. * class. </p></li>
  17. *
  18. * </ol>
  19. *
  20. * <p> If the class was found using the above steps, and the
  21. * <tt>resolve</tt> flag is true, this method will then invoke the {@link
  22. * #resolveClass(Class)} method on the resulting <tt>Class</tt> object.
  23. *
  24. * <p> Subclasses of <tt>ClassLoader</tt> are encouraged to override {@link
  25. * #findClass(String)}, rather than this method. </p>
  26. *
  27. * <p> Unless overridden, this method synchronizes on the result of
  28. * {@link #getClassLoadingLock <tt>getClassLoadingLock</tt>} method
  29. * during the entire class loading process.
  30. *
  31. * @param name
  32. * The <a href="#name">binary name</a> of the class
  33. *
  34. * @param resolve
  35. * If <tt>true</tt> then resolve the class
  36. *
  37. * @return The resulting <tt>Class</tt> object
  38. *
  39. * @throws ClassNotFoundException
  40. * If the class could not be found
  41. */
  42. protected Class<?> loadClass(String name, boolean resolve)
  43. throws ClassNotFoundException
  44. {
  45. synchronized (getClassLoadingLock(name)) {
  46. // First, check if the class has already been loaded
  47. Class<?> c = findLoadedClass(name);
  48. if (c == null) {
  49. long t0 = System.nanoTime();
  50. try {
  51. if (parent != null) {
  52. c = parent.loadClass(name, false);
  53. } else {
  54. c = findBootstrapClassOrNull(name);
  55. }
  56. } catch (ClassNotFoundException e) {
  57. // ClassNotFoundException thrown if class not found
  58. // from the non-null parent class loader
  59. }
  60.  
  61. if (c == null) {
  62. // If still not found, then invoke findClass in order
  63. // to find the class.
  64. long t1 = System.nanoTime();
  65. c = findClass(name);
  66.  
  67. // this is the defining class loader; record the stats
  68. sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
  69. sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
  70. sun.misc.PerfCounter.getFindClasses().increment();
  71. }
  72. }
  73. if (resolve) {
  74. resolveClass(c);
  75. }
  76. return c;
  77. }
  78. }

破坏双亲委派模型

双亲委派模型虽然在类加载器中很重要,但是并不是Java强制性要求的一个模型,而是Java设计者推荐给开发者的类加载器的实现方式。在Java世界中大部分的类加载器都遵循这个模型,但也有例外,双亲委派模型到目前为止主要出现过3次较大规模的“被破坏”情况。

  • 第一次:由于类加载器是JDK1.0就已经存在了,而双亲委派模型在JDK1.2之后才被引入,所以为了向前兼容,做了一些妥协。在JDK1.2以后已不再提倡用户去覆盖loadClass()方法,而应该把自己的实现逻辑写在findClass()方法中,这样在loadClass方法中如果父类加载器加载失败,就会调用自己的findClass方法来完成加载,这样就保证了自己实现的类加载器符合双亲委派模型了。
  • 第二次:双亲委派模型的规则是自低向上(由子到父)来进行加载的,但是有些情况下父类是需要调用子类的代码,这种情况就需要破坏这个模型了。为了解决这种情况,Java设计团队,引入了一个新的加载器:线程上下文加载器(Trhead Context ClassLoader)。这个类加载器可以通过java.lang.Thread类的setCOntextClassLoader()方法进行设置,通过getContextClassLoader()方法来获得。如果创建线程时还未设置,它会从父线程中继承一个,如果在应用程序的全局范围内都没有设置过的话,那这个类加载器就是应用程序类加载器了。Java中所有涉及SPI的加载动作基本上都采用这种方式,例如:JNDI、JDBC、JCE、JAXB、和JBI等。其实我们常用的Tomcat这种应用服务器也是使用的这种类加载器。
  • 第三次:为了实现热部署,热插拔,模块化等功能。就是说更新了一些模块而不需要重启,只需要把类和类加载器一同替换掉就可以实现热部署了。

JVM学习记录-类加载器的更多相关文章

  1. Java虚拟机JVM学习05 类加载器的父委托机制

    Java虚拟机JVM学习05 类加载器的父委托机制 类加载器 类加载器用来把类加载到Java虚拟机中. 类加载器的类型 有两种类型的类加载器: 1.JVM自带的加载器: 根类加载器(Bootstrap ...

  2. JVM学习--(六)类加载器原理

    我们知道我们编写的java代码,会经过编译器编译成字节码文件(class文件),再把字节码文件装载到JVM中,映射到各个内存区域中,我们的程序就可以在内存中运行了.那么字节码文件是怎样装载到JVM中的 ...

  3. JVM学习记录-类加载的过程

    类的整个生命周期的7个阶段是:加载(Loading).验证(Verification).准备(Preparation).解析(Resolution).初始化(Initialization).使用(Us ...

  4. JVM学习笔记——类加载器与类加载过程

    类加载器与类加载过程 类加载器ClassLoader 类加载器 ClassLoader 用于把 class 文件装载进内存. 启动类加载器(Bootstrap ClassLoader): 这个类加载使 ...

  5. JVM学习记录-垃圾收集器

    先回顾一下上一篇介绍的JVM中常见几种垃圾收集算法: 标记-清除算法(Mark-Sweep). 复制算法(Copying). 标记整理算法(Mark-Compact). 分代收集算法(Generati ...

  6. JVM学习记录-类加载时机

    虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是类的加载机制. 在Java语言里面,类型的加载.连接和初始化过程都 ...

  7. JVM学习笔记——类加载和字节码技术篇

    JVM学习笔记--类加载和字节码技术篇 在本系列内容中我们会对JVM做一个系统的学习,本片将会介绍JVM的类加载和字节码技术部分 我们会分为以下几部分进行介绍: 类文件结构 字节码指令 编译期处理 类 ...

  8. JVM学习笔记——类加载过程

    JVM学习笔记——类加载过程 类加载模型——双亲委派模型(Parents Delegation Model)也可称为“溯源委派加载模型” Java的类加载器是一个运行时核心基础设施模块,主要是启动之初 ...

  9. JVM的艺术—类加载器篇(二)

    分享是价值的传递,喜欢就点个赞 引言 今天我们继续来深入的剖析类加载器的内容.上节课我们讲了类加载器的基本内容,没看过的小伙伴请加关注.今天我们继续. 什么是定义类加载器和初始化类加载器? 定义类加载 ...

随机推荐

  1. iOS处理视图上同时添加单击与双击手势的冲突问题

    _bgView.userInteractionEnabled = YES; //在cell上添加 bgView,给bgView添加两个手势检测方法 UITapGestureRecognizer *do ...

  2. 关于使用java自动发邮件--找不到smtphost

    今天解决报的第二个异常:Unknown SMTP host: smtp.qq.com;在找了网上的一些资料后,有看到是因为使用了代理服务器,所以无法访问.我试着用了telnet去访问,确实不行.最近都 ...

  3. iOS 百度地图截屏

    关于百度地图截屏的问题,发现不能用常用的方法进行载屏,常用的截屏方法所得到的图片地图瓦片底图会显示空白,网上给出的答案是这样的 :因为百度地图不是用UIKit实现的,所以得不到截图! 不过通过Open ...

  4. fping常用参数介绍

    fping的主要参数有以下两个: -a:只显示存活主机: -u:只显示不存活主机: -l:循环ping目标IP地址的输入方式: fping IP1 IP2 IP3 ...: fping -f file ...

  5. ServiceStack 多租户的实现方案

    以SqlServer为例子说明ServiceStack实现多租户,在SqlServer中创建4个Database:TMaster.T1,T2,T3,为了安全起见 每个Database不用sa账号,而是 ...

  6. Log4Net记录到数据库

    WinForm下Log4Net的配置 Log4Net 组件下载地址 https://download.csdn.net/download/zgx123zgx123zg/10470986 configS ...

  7. 拖拽TreeViewItem到OCX控件

    由于C#在性能方面,和C++还是有不少的差距,所以在项目中有一块是用C++的OCX控件实现,然后包括在WPF项目中.由于C++,C#属于不同的体系架构,造成了许多问题,特使是拖拽TreeViewIte ...

  8. C# MemoryStream BinaryReader

    不清楚这类东西内部搞什么鬼,直接看代码才舒爽 https://referencesource.microsoft.com/#mscorlib 然后可以在线测试 https://www.tutorial ...

  9. java urlrewrite实现伪静态化

    1.示例 http://www.onlyfun.com/goods/company.jsp?companyId=455326 ==> http://www.onlyfun.com/company ...

  10. (原创)Callable、FutureTask中阻塞超时返回的坑点

    直接上代码 import java.util.concurrent.Callable; public class MyCallable implements Callable<String> ...