|------Bootstrap 启动加载器
+--- 主要负责加载 jdk/lib目录下的核心api 或者 -Xbootclasspath选项指定的jar包
|------Extension 扩展加载器
+--- 主要负责加载 jdk/lib/ext目录下的核心api 或者 -Djava.ext.dirs 指定目录下jar包
|------System 系统加载器
+--- 主要负责加载 java -classpath/-Djava.class.path所指定目录下的类
|------Custom ClassLoader 自定义类加载器 {java.lang.ClassLoader的子类}
+--- 程序运行期间,通过其动态加载class .
|------每个Classloader都维护一份自己的名称空间,同一个空间中,不能出现两个同名的类。
|------Java类加载器采用“双亲委托的加载机制”
下面,我们来看一个自定义类加载器的处理流程:
以下代码,演示除了bootstarp加载器外的类加载器加载流程:
// 检查类是否已被装载过 Class c = findLoadedClass(name); if (c == null ) { // 指定类未被装载过 try { if (parent != null ) { // 如果父类加载器不为空, 则委派给父类加载 c = parent.loadClass(name, false ); } else { // 如果父类加载器为空, 则委派给启动类加载加载 c = findBootstrapClass0(name); } } catch (ClassNotFoundException e) { // 启动类加载器或父类加载器抛出异常后, 当前类加载器将其 // 捕获, 并通过findClass方法, 由自身加载 c = findClass(name); } }
利用Class.forName来加载类,Class.forName使用的是被调用者的类加载器来加载类的.
这种特性,证明了java类加载器中的名称是唯一的,不会互相干扰,既在一般环境中,保证同一类与之关联的其他类都是通过当前类的类加载器所加载的。
public static Class forName(String className) throws ClassNotFoundException { return forName0(className, true , ClassLoader.getCallerClassLoader());
//getCallerClassLoader() 就是得到调用当前forName方法的类的类加载器 } /** Called after security checks have been made. */ private static native Class forName0(String name, boolean initialize, ClassLoader loader) throws ClassNotFoundException;
线程上下文类加载器:
Java默认的线程上下文类加载器是 系统类加载器(AppClassLoader)
// Now create the class loader to use to launch the application try { loader = AppClassLoader.getAppClassLoader(extcl); } catch (IOException e) { throw new InternalError( "Could not create application class loader" ); } // Also set the context class loader for the primordial thread. Thread.currentThread().setContextClassLoader(loader);
在sun.misc.Launch的无参构造函数Launch()中定义了上述代码。
使用线程上下文加载器,可以在执行线程中,抛弃双亲委托加载链模式,使用线程上下文里的类加载器加载类。
如: 在线程上下文中加载第三方的JNDI实现,而不依赖于双亲委托。
自定义类加载器: