双亲委派机制在我们平常的面试过程中经常被问到,下面给大家详细讲解双亲委派机制及作用
一、概念
当一个类收到了类加载请求,他首先不会尝试自己去加载这个类,而是把这个请求委派给父类去完成,只有当父类加载器反馈自己无法完成这个请求的时候,子类加载器才会尝试自己去加载
二、类别
自定义加载器(CustomerClassLoader)
java编写,用户自定义的类加载器,可加载指定路径下的class文件。在Tomcat中,就自定义了一套自定义的加载器,因为Tomcat可部署多个web应用,若是在不同的web应用中含有相同的路径class,则会出现加载异常,所以Tomcat自定义自己的加载器,保证加载正确
系统类加载器(AppClassLoader)
java编写,加载程序所在的目录,如在user.dir所在的位置的class
标准拓展类加载器(ExtClassLoader)
java编写,加载扩展库。如加载classpath中的jre,javax.*或者一些第三方jar包中指定位置中的类,
启动类加载器(BootstapClassLoader)
c++编写,加载java核心库java.*等一些jdk自带的java类。如String等
三、源码分析
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// 首先检查这个classsh是否已经加载过了
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
// c==null表示没有加载,如果有父类的加载器则让父类加载器加载
if (parent != null) {
c = parent.loadClass(name, false);
} else {
//如果父类的加载器为空 则说明递归到bootStrapClassloader了
//bootStrapClassloader比较特殊无法通过get获取
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {}
if (c == null) {
//如果bootstrapClassLoader 仍然没有加载过,则递归回来,尝试自己去加载class
long t1 = System.nanoTime();
c = findClass(name);
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
四、作用
1.防止重复加载同一个class。通过委托确认是否加载,如已加载,无需重复加载,保证数据安全
2.防止核心class被修改,如String类,若我们自定义一个java.lang.String类,则不会去加载,会加载jdk自带的String类
五、总结
希望能帮助到大家,谢谢