类与类加载器(双亲委派模型)

1、类和类加载器

只是用于实现类的加载动作
类加载器分为:
启动类加载器(bootstrap class loader),负责加载JRE核心类库,如lib目录下rt.jar、tools.jar等
扩展类加载器(extension class loader),负责加载ext目录中的类库
引用程序类加载器(application class loader),负责加载用户类路径上所有类库
自定义加载器(user class loader)非必须,用户自己开发的类加载器,加载类的方式用户自定义

2、双亲委派模型

类加载器主要遵循双亲委派模型模式建立

上图说明:当一个Hello.class这样的文件要被加载时。不考虑我们自定义类加载器,首先会在AppClassLoader中检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理也会先检查自己是否已经加载过,如果没有再往上。注意这个类似递归的过程,直到到达Bootstrap classLoader之前,都是在检查是否加载过,并不会选择自己去加载。直到BootstrapClassLoader,已经没有父加载器了,这时候开始考虑自己是否能加载了,如果自己无法加载,会下沉到子加载器去加载,一直到最底层,如果没有任何加载器能加载,就会抛出ClassNotFoundException

3、加载类主要代码(源码)

public abstract class ClassLoader {
  protected Class <? > loadClass(String name, boolean resolve) throws ClassNotFoundException {
    synchronized(getClassLoadingLock(name)) {
      // 首先,检查类是否已经加载
      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) {
          // ClassNotFoundException thrown if class not found
          // from the non-null parent class loader
        }
        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;
    }
  }
}
posted @ 2021-07-04 18:32  liuqIT  阅读(67)  评论(0)    收藏  举报