打破Java中类加载器可见性原则的代码解析
在Java中,类加载器遵守可见性原则,即一个类加载器只能加载其父类加载器或自身加载过的类和资源。然而,在某些情况下,这一原则会被打破。
如示例代码所示,当tomcat加载spring-web模块时,它使用ServiceLoader加载所有实现了ServletContainerInitializer接口的类。这会违背可见性原则,因为这些类可能是由其他类加载器加载的。
该代码使用Thread.currentThread().getContextClassLoader()方法获取当前线程的上下文类加载器,该加载器实际上是sun.misc.Launcher.AppClassLoader。这个类加载器不是bootstrapClassLoader或其直接子加载器,它负责加载位于classPath下的类。
立即学习“Java免费学习笔记(深入)”;
相反,正常的双亲委派过程如下所示:
- 由BootstrapClassLoader加载rt.jar中的接口类。
- AppClassLoader寻找接口的实现类,但无法在classPath中找到。
- AppClassLoader无法进一步委托给父加载器,因为它就是最顶层的加载器。
- 因此,AppClassLoader必须指定一个线程上下文类加载器(sun.misc.Launcher.AppClassLoader),该加载器可以加载外部jar包中的实现类。
因此,通过设置线程上下文类加载器,ServiceLoader可以打破可见性原则,允许加载由其他类加载器加载的类。