最近和朋友聊到了双亲委派、classloader相关的内容,秉持着查漏补缺的心态,对这里的知识做一个总结和归纳。这里仅仅是java 8 ,因为大体了解 java 9 以后引模块的概念,有点不一样。错误的地方希望能够得到指正。
什么是ClassLoader
通过API 文档我们能看到,Classloader类的主要作用就是加载类的,这是废话。
在JVM规范 中的意思也是如此,就是把类,也就是class文件加载到jvm中去。
类的加载器
Java自带了三个类加载器如下
- Bootstrap ClassLoader
- Extention ClassLoader
- Appclass ClassLoader
其中Bootstrap ClassLoader,是C语言实现的,剩下的两个是在sun.misc.Launcher中的两个静态类,有兴趣的可以自己去看看。
在启动Main方法的时候可以加入 -XX:+TraceClassLoading参数,查看加载的顺序
Bootstrap ClassLoader
这个是jar 运行环境自带的,主要加载System.getProperty(“sun.boot.class.path”) 路径
Extention ClassLoader
这个可以通过java的代码看到System.getProperty(“java.ext.dirs”)
如下代码是ExtClassLoader中的
private static File[] getExtDirs() {
String var0 = System.getProperty("java.ext.dirs");
File[] var1;
if (var0 != null) {
StringTokenizer var2 = new StringTokenizer(var0, File.pathSeparator);
int var3 = var2.countTokens();
var1 = new File[var3];
for(int var4 = 0; var4 < var3; ++var4) {
var1[var4] = new File(var2.nextToken());
}
} else {
var1 = new File[0];
}
return var1;
}
Appclass ClassLoader
通过代码可以看到这个是== System.getProperty(“java.class.path”);==
public static ClassLoader getAppClassLoader(final ClassLoader var0) throws IOException {
final String var1 = System.getProperty("java.class.path");
final File[] var2 = var1 == null ? new File[0] : Launcher.getClassPath(var1);
return (ClassLoader)AccessController.doPrivileged(new PrivilegedAction<Launcher.AppClassLoader>() {
public Launcher.AppClassLoader run() {
URL[] var1x = var1 == null ? new URL[0] : Launcher.pathToURLs(var2);
return new Launcher.AppClassLoader(var1x, var0);
}
});
}
JVM主要的类加载机制
父级委派(双亲委派)
也不知道谁起的名字,很明显有歧义。我觉得就应该叫父级委派。说白了就是一个定义,所有的类都要委托到上层的加载器加载,如果上层没有,再有下层加载器加载。