一、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. java工作流软件发送邮件的方案

    利用javamail的功能将发送邮件的功能集成到java工作流系统中.javamail包提供有发送邮件的方法,设置发送人地址,收件人地址,抄送,主题,邮件服务器地址,认证用户等信息,再调用javama ...

  2. variably modified 'dist' at file scope|

    转自:http://blog.csdn.net/wusuopubupt/article/details/18408227 错误原因: The reason for this warning is th ...

  3. Nopcommerce 二次开发1 基础

    1  Doamin    酒店 namespace Nop.Core.Domain.Hotels { /// <summary> /// 酒店 /// </summary> p ...

  4. 软件工程—我的IT的新篇章

    这学期开了一门软件工程,在开始学习过程中,我对软件工程充满了浓厚的兴趣,我很享受软件工程带给我的快乐与激情,让我对计算机编程更加的了解与享受,我非常的喜欢这门学科,我对老师的教学方式非常的喜欢,我觉得 ...

  5. CString std::string相互转换

    CString->std::string 例子: CString strMfc=“test“; std::string strStl; strStl=strMfc.GetBuffer(0); s ...

  6. java remote debug parameters

    java -agentlib:jdwp=transport=dt_socket,server=y,address=8000,suspend=n

  7. 在CentOS或RHEL防火墙上开启端口

    转载自:https://linux.cn/article-4243-1.html 如果希望在服务器上提供服务,诸如CentOS或RHEL的企业级Linux发行版包含内置的强大防火墙,它们默认的防火墙规 ...

  8. LR12.53—第3课:重播Vuser脚本

    第3课:重播Vuser脚本 在前面的教训,你记录了一组典型的用户行为,并准备重播脚本. 重播脚本之前,您可以配置脚本的运行时设置,它定义了Vuser的行为. 注:记录与基于Web的协议的地方WebTo ...

  9. THINKPHP中关于接口问题(客户端)

    一 apk版本号 客户端发送get请求访问服务器端的控制器方法,通过用户传过来的用户名和密码. 二  服务器端通过客户端传入的user and password 去数据库进行查询,Success就生成 ...

  10. 计算机缺失缺少mfc110.dll等相关文件的解决办法

    去https://www.microsoft.com/zh-CN/download/details.aspx?id=30679下载 VSU4\vcredist_x64.exe 和VSU4\vcredi ...