一、Java中的class加载机制有以下三个特性:

1、全盘负责制 

“全盘负责”是指当一个ClassLoader装载一个类时,除非显示地使用另一个ClassLoader,则该类所依赖及引用的类也由这个CladdLoader载入。

例如,系统类加载器AppClassLoader加载入口类(含有main方法的类)时,会把main方法所依赖的类及引用的类也载入,依此类推。“全盘负责”机制也可称为当前类加载器负责机制。显然,入口类所依赖的类及引用的类的当前类加载器就是入口类的类加载器。

2、双亲委派制(Parent Delegation)

1) 委托机制的意义

主要是出于安全性考虑,确保Java的核心类在内存中只有一份字节码,比如两个类A和类B都要加载java.lang.System类,通过双亲委派,系统只会加载一次java.lang.System,即使用户重写了java.lang.System,也不会有机会被加载到,除非你重写ClassLoader。不过有时候为了做容器隔离,需要在JVM中对同一个Class有多份字节码,例如OSGI和Pandora技术,后面会详细谈到。

2) 委托机制是必须的吗?

“双亲委派”机制只是Java推荐的机制,并不是强制的机制。我们可以继承java.lang.ClassLoader类,实现自己的类加载器。如果想保持双亲委派模型,就应该重写findClass(name)方法;如果想破坏双亲委派模型,可以重写loadClass(name)方法。使用自己的类加载器,有很多高级玩法,例如OSGI和Pandora的隔离机制,就是通过自定义ClassLoader来实现的。

3) 如何实现双亲委派?

  默认的ClassLoader的loadClass()实现方式是双亲委派模型,我们可以继承ClassLoader去自定义自己的ClassLoader,如果不重写loadClass方法,那么默认也是双亲委派的。例如,URLClassLoader只是实现了findClass( ),而loadClass( )还是继承ClassLoader的,所以其依然是Parent Delegation的。下面是ClassLoader.loadClass( )源码,看下双亲委派是怎么实现的。

1)Check这个class是否被装载过,如果有直接返回。

装载过的类是被缓存起来的,这样确保了同一个类不会被加载两次,不过有一个问题,用什么来作为class缓存的key呢?在JVM中,class是用类全名(包名+类名) 再加上加载这个类的classLoader来唯一标识的,例如class的类全名是C1,classLoader是L1,那么这个class instance在JVM中的key就是(C1, L1),此时另一个classLoader L2也加载了该类,那么将会有另一个class instance (C1, L2),这两个class instance是不同的type,如果这两个class的object做赋值操作的话,会出现ClassCastException。

2)尝试从parent classloader去加载类。

3)如果parent是null(当parent是bootstrap时就是null了),试图从BootstrapClassLoader的native方法去加载类。

4)在上面尝试都失败的情况下,尝试自己去加载。

3、按需加载 (On-demand Loading)  

什么时候Class会被JVM加载呢? 回答是只有当class被用到时,才会被load,例如new instance,调用其static变量和方法,或使用反射调用其class对象。

这个很容易验证,在启动参数里加上-verbose:class,  就可以清晰看到class是何时被加载的。

二、JVM中classloader加载class的顺序

三、ContextClassloader的用处

1)什么是ContextClassLoader

Thread的一个属性,可以在运行时,通过setContextClassLoader方法来指定一个合适的classloader作为这个线程的contextClassLoader,然后在任何地方通过getContextClassLoader方法来获得此contextClassLoader,用它载入我们所需要的Class。如果没有被显示set过,默认是system classloader。利用这个特性,我们可以“打破”classloader委托机制,父classloader可以获得当前线程的contextClassLoader,而这个contextClassLoader可以是它的子classloader或者其他的classloader。

2) 为什么要使用ContextClassLoader

Thread context classloaders provide a back door around the classloading delegation scheme.

Take JNDI for instance: its guts are implemented by bootstrap classes in rt.jar (starting with J2SE 1.3), but these core JNDI classes may load JNDI providers implemented by independent vendors and potentially deployed in the application's -classpath. This scenario calls for a parent classloader (the primordial one in this case) to load a class visible to one of its child classloaders (the system one, for example). Normal J2SE delegation does not work, and the workaround is to make the core JNDI classes use thread context loaders, thus effectively "tunneling" through the classloader hierarchy in the direction opposite to the proper delegation.

这个后门在SPI的实现中很有用,因为接口类是在parent classloader中加载的,而实现类是由它的child classloader加载的,使用contextClassLoader可以绕过双亲委派,达到在parent中使用child classloader去load class的目的。其过程如下图所示:

这样的示例在JDK中很常见,例如JNDI和JAXP都是通过这样的方式去加载具体的provider的。例如

javax.xml.ws.spi.FactoryFinder
Object find(String factoryId, String fallbackClassName){
ClassLoader classLoader;
try {
// get context classloader, mostly it's system classloader, but it could be user-defined classloader as well.
classLoader = Thread.currentThread().getContextClassLoader();
} catch (Exception x) {
throw new WebServiceException(x.toString(), x);
}
String serviceId = "META-INF/services/" + factoryId;
// try to find services in CLASSPATH
// Note that if it's not system classloader, this will invoke user-defined classloader's findResource( ) to find services when all its parents failed.
try {
InputStream is=null;
if (classLoader == null) {
is=ClassLoader.getSystemResourceAsStream(serviceId);
} else {
is=classLoader.getResourceAsStream(serviceId);
}
.....
return newInstance(fallbackClassName, classLoader);
}

Java Classloader详解的更多相关文章

  1. Java ClassLoader详解(转载)

    Java ClassLoader详解 类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK ...

  2. (转)Java ClassLoader详解

    转:http://java.chinaitlab.com/base/804400.html 类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态 ...

  3. Java ClassLoad详解

    Java ClassLoad详解 类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK 1. ...

  4. Java内部类详解

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

  5. 黑马----JAVA迭代器详解

    JAVA迭代器详解 1.Interable.Iterator和ListIterator 1)迭代器生成接口Interable,用于生成一个具体迭代器 public interface Iterable ...

  6. C++调用JAVA方法详解

    C++调用JAVA方法详解          博客分类: 本文主要参考http://tech.ccidnet.com/art/1081/20050413/237901_1.html 上的文章. C++ ...

  7. Java虚拟机详解----JVM常见问题总结

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...

  8. [转] Java内部类详解

    作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置 ...

  9. Java面向对象详解

    Java面向对象详解 前言:接触项目开发也有很长一段时间了,最近开始萌发出想回过头来写写以前学 过的基础知识的想法.一是原来刚开始学习接触编程,一个人跌跌撞撞摸索着往前走,初学的时候很多东西理解的也懵 ...

随机推荐

  1. C# 报表设计器 (winform 设计端)开发与实现生成网页的HTML报表

    记得2010年之前,公司的项目基本上都要用到报表,以前我们常用的方法就是针对客户的需求来定制化开发(基本上是死写代码)来实现,经常导致项目经常性的延期,因为客户的需求经常会变化,随着用户的使用认知度的 ...

  2. Android SQLiteOpenHelper类的使用

    SQLiteOpenHelper类是Android平台提供的用于SQLite数据库的创建.打开以及版本管理的帮助类.一般需要继承并这个类并实现它的onCreate和onUpgrade方法,在构造方法中 ...

  3. protocol

    For every object that can have a delegate, there is a corresponding protocol that declares themessag ...

  4. C连接MySQL数据库开发之Windows环境配置及测试(转)

    http://blog.csdn.net/xyang81/article/details/26814633(转)

  5. 《疯狂Java讲义》(四)---- 面向对象&基于对象

    "基于对象"也使用了对象,但是无法利用现有的对象模板产生新的对象类型,继而产生新的对象,也就是说,"基于对象"没有继承的特点,而多态更需要继承,所以" ...

  6. iOS-多线程 ,整理集锦,多种线程的创建

    - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launc ...

  7. python中logging

    1.root logger以及logger斧子关系    http://www.pythonclub.org/modules/logging #coding=utf-8 __author__ = 'n ...

  8. js获取项目根目录的方法

    getRootPath = function(){ //获取当前网址,如: http://localhost:8080/ems/Pages/Basic/Person.jsp var curWwwPat ...

  9. 完全卸载Oracle11G

    要特别注意删除注册表的这块,如果删错了会导致系统出现问题,而且oracle的安装卸载真的很烦,一旦装错了,卸载不干净就会导致种种的问题无法再次安装,个人建议用360卸载,360卸载完成后会自动检测到无 ...

  10. Linux Runlevel 启动 脚本

    Linux 操作系统自从开始启动至启动完毕需要经历几个不同的阶段,这几个阶段就叫做 Runlevel,同样,当Linux操作系统关闭时也要经历另外几个不同的 Runlevel,下面详细介绍一下 Run ...