线程上下文类加载器(context class loader)是从JDK 1.2开始引入的。类 java.lang.Thread中的方法getContextClassLoader()和setContextClassLoader(ClassLoader cl)用来获取和设置线程的上下文类加载器。如果没有通过 setContextClassLoader(ClassLoader cl)方法进行设置的话,线程将继承其父线程的上下文类加载器。Java 应用运行的初始线程的上下文类加载器是系统类加载器。在线程中运行的代码可以通过此类加载器来加载类和资源。
为了加载类,Java还提供了很多服务提供者接口(Service Provider Interface,SPI),允许第三方为这些接口提供实现。那类加载就会存在问题:SPI 的接口是 Java 核心库的一部分,是由引导类加载器来加载的;SPI 实现的 Java 类一般是由系统类加载器来加载的。引导类加载器是无法找到 SPI 的实现类的,因为它只加载 Java 的核心库,正常来说,在java 核心库里的代码都是需要用引导类加载器 来加载的,但是 有个例外class.forName("子类实现"),引导类加载器执行这句代码的时候,必然加载 子类实现类,但是它并加载不了,怎么办,只能通过线程上下文类加载器,,于是代码就改成了 class.forName("子类实现",false,threadcontextloader).这样就可以了。
在 SPI 接口的代码中使用线程上下文类加载器,就可以成功的加载到 SPI 实现的类
我们拿jdbc为例子分析:
private DriverManager(){}
-
/**
-
* Load the initial JDBC drivers by checking the System property
-
* jdbc.properties and then use the {@code ServiceLoader} mechanism
-
*/
-
static {
-
loadInitialDrivers();
-
println("JDBC DriverManager initialized");
-
}
-
private static void loadInitialDrivers() {
-
AccessController.doPrivileged(new PrivilegedAction<Void>() {
-
//省略...
-
public Void run() {
-
ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
-
Iterator driversIterator = loadedDrivers.iterator();
-
//省略...
-
return null;
-
}
-
});
-
}
-
}
其中ServiceLoader.load(Driver.class)方法源码:
public static <S> ServiceLoader<S> load(Class<S> service) {
-
ClassLoader cl = Thread.currentThread().getContextClassLoader();
-
return ServiceLoader.load(service, cl);
-
}
可见jdbc在通过SPI方式加载service实现类的时候是使用的线程上下文加载器加载的