文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java源码解析之ClassLoader

2024-04-02 19:55

关注

一、前言

一个完整的Java应用程序,当程序在运行时,即会调用该程序的一个入口函数来调用系统的相关功能,而这些功能都被封装在不同的class文件当中,所以经常要从这个class文件中要调用另外一个class文件中的方法,如果另外一个文件不存在的,则会引发系统异常。而程序在启动的时候,并不会一次性加载程序所要用的所有class文件,而是根据程序的需要,通过Java的类加载机制(ClassLoader)来动态加载某个class文件到内存当中的,从而只有class文件被载入到了内存之后,才能被其它class所引用。所以ClassLoader就是用来动态加载class文件到内存当中用的。

Android平台上虚拟机运行的是Dex字节码,一种对class文件优化的产物,传统Class文件是一个Java源码文件会生成一个.class文件,而Android是把所有Class文件进行合并,优化,然后生成一个最终的class.dex,目的是把不同class文件重复的东西只需保留一份,如果我们的Android应用不进行分dex处理,最后一个应用的apk只会有一个dex文件。

二、java 中的 ClassLoader

BootstrapClassLoader
负责加载 JVM 运行时的核心类,比如 JAVA_HOME/lib/rt.jar 等等

ExtensionClassLoader
负责加载 JVM 的扩展类,比如 JAVA_HOME/lib/ext 下面的 jar 包

AppClassLoader
负责加载 classpath 里的 jar 包和目录

三、Android 中的 ClassLoader

BootClassLoader

负责 Android系统启动时会使用BootClassLoader来预加载常用类,与Java中的Bootstrap ClassLoader不同的是,它并不是由C/C++代码实现,而是由Java实现的。BootClassLoader是ClassLoader的一个内部类。
PathClassLoader

负责加载已经安装的Apk,也就是/data/app/package 下的apk文件,也可以加载/vendor/lib, /system/lib下的nativeLibrary

DexClassLoader

负责加载可以加载一个未安装的apk文件。

四、双亲委派机制

每一个 ClassLoader 中都有一个 parent 对象,代表的是父类加载器,在加载一个类的时候,会先使用父类加载器去加载,如果在父类加载器中没有找到,自己再进行加载,如果 parent 为空,那么就用系统类加载器来加载。通过这样的机制可以保证系统类都是由系统类加载器加载的。 下面是 ClassLoader 的 loadClass 方法的具体实现。


protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                try {
                    if (parent != null) {
                        // 先从父类加载器中进行加载
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
 
                if (c == null) {
                    // 没有找到,再自己加载
                    c = findClass(name);
                }
            }
            return c;
    }

五、源码分析

1.现在我们看下 BaseDexClassLoader 继承自ClassLoader


public class BaseDexClassLoader extends ClassLoader{
 
	...
	
	//存放需要加载的dexList
	private final DexPathList pathList;
	
	
    public BaseDexClassLoader(String dexPath, File optimizedDirectory,
            String librarySearchPath, ClassLoader parent) {
        this(dexPath, optimizedDirectory, librarySearchPath, parent, false);
    }
 
	
    public BaseDexClassLoader(String dexPath, File optimizedDirectory,
            String librarySearchPath, ClassLoader parent, boolean isTrusted) {
        super(parent);
        this.pathList = new DexPathList(this, dexPath, librarySearchPath, null, isTrusted);
 
        ...
    }
	
	 
	public BaseDexClassLoader(ByteBuffer[] dexFiles, ClassLoader parent) {
        // TODO We should support giving this a library search path maybe.
        super(parent);
        this.pathList = new DexPathList(this, dexFiles);
    }
	
	
	@Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        List<Throwable> suppressedExceptions = new ArrayList<Throwable>();
		//1 在pathList中寻找name对应的类
        Class c = pathList.findClass(name, suppressedExceptions);
		// 如果未找到此类,则抛出异常
        if (c == null) {
            ClassNotFoundException cnfe = new ClassNotFoundException(
                    "Didn't find class \"" + name + "\" on path: " + pathList);
            for (Throwable t : suppressedExceptions) {
                cnfe.addSuppressed(t);
            }
            throw cnfe;
        }
        return c;
    }
}

PathClassLoader 和DexClassLoader: 继承自BaseDexClassLoader


public class PathClassLoader extends BaseDexClassLoader {
  
     
    public PathClassLoader(String dexPath, ClassLoader parent) {
        //调用父类BaseDexClassLoader 四参构造方法
        super(dexPath, null, null, parent);
    }
 
    
    public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {
        //调用父类BaseDexClassLoader 四参构造方法
        super(dexPath, null, librarySearchPath, parent);
    }
}


public class DexClassLoader extends BaseDexClassLoader {
 
	 
    public DexClassLoader(String dexPath, String optimizedDirectory,
            String librarySearchPath, ClassLoader parent) {
			
		// 调用父类BaseDexClassLoader 四参构造方法,在API26以上,librarySearchPath参数已弃用,使用此方法		
        super(dexPath, null, librarySearchPath, parent);
		
		// API26及以下使用此方法		
		// super(dexPath, new File(optimizedDirectory), librarySearchPath, parent);
    }
}

我们看1 处 pathList 的 findClass 是如何查找的


 final class DexPathList {
    private static final String DEX_SUFFIX = ".dex";
    private static final String zipSeparator = "!/";
 
    private final ClassLoader definingContext;
 
	// dex/resource 存放dex的数组
    private Element[] dexElements;
 
	// 存放本地库文件的列表
    private final List<File> nativeLibraryDirectories;
 
	// 存放系统本地库文件的列表
    private final List<File> systemNativeLibraryDirectories;
 
	// 存放创建dexElement列表时引发异常的列表
    private IOException[] dexElementsSuppressedExceptions;
	
	DexPathList(ClassLoader definingContext, String dexPath,
            String librarySearchPath, File optimizedDirectory, boolean isTrusted) {
        ...
 
        this.definingContext = definingContext; //BaseDexClassLoader构造器中会传入其本身
 
        ArrayList<IOException> suppressedExceptions = new ArrayList<IOException>();
        // 通过dexPath路径使用分隔符将其转换成dexElements列表
        this.dexElements = makeDexElements(splitDexPath(dexPath), optimizedDirectory,
                                           suppressedExceptions, definingContext, isTrusted);
 
       
        ...
    }
	
	 // 为本地库搜索路径生成一个directory/zip path元素数组
	 private static Element[] makeDexElements(List<File> files, File optimizedDirectory,
            List<IOException> suppressedExceptions, ClassLoader loader, boolean isTrusted) {
      Element[] elements = new Element[files.size()];
      int elementsPos = 0;
     
      for (File file : files) {
          if (file.isDirectory()) { //如果是文件夹,则直接添加至elements
              elements[elementsPos++] = new Element(file);
          } else if (file.isFile()) { //如果是文件
       
              String name = file.getName();
              DexFile dex = null;
              //是否为 .dex 文件
              if (name.endsWith(DEX_SUFFIX)) {
                  // Raw dex file (not inside a zip/jar).
                  try {
                      //如果是 dex 文件,则加载这个文件
                      dex = loadDexFile(file, optimizedDirectory, loader, elements);
                      if (dex != null) {
                          //将dex 文件保存,注意第二个参数传的底 null 
                          elements[elementsPos++] = new Element(dex, null);
                      }
                  } catch (IOException suppressed) {
                      System.logE("Unable to load dex file: " + file, suppressed);
                      suppressedExceptions.add(suppressed);
                  }
              } else {
                  //如果不是目录且不是 .dex 结尾的,那么他可能是 jar。加载这个文件
                      dex = loadDexFile(file, optimizedDirectory, loader, elements);          
                  if (dex == null) {
                      elements[elementsPos++] = new Element(file);
                  } else {
                      // 保存dex 文件 和 当前的 file,这种情况下可能是 jar,具体可以看这个构造方法
                      elements[elementsPos++] = new Element(dex, file);
                  }
 
              if (dex != null && isTrusted) { //如果dex对象不为空且是允许信任状态
                dex.setTrusted(); // 将此dex对象设置为已信任,它可以访问平台的隐藏api
              }
          } else {
              System.logW("ClassLoader referenced unknown path: " + file);
          }
      }
      if (elementsPos != elements.length) {
          elements = Arrays.copyOf(elements, elementsPos);
      }
      return elements;
    }
	
	public Class<?> findClass(String name, List<Throwable> suppressed) {
		// 遍历dex列表
        for (Element element : dexElements) {
            //2   
            Class<?> clazz = element.findClass(name, definingContext, suppressed);
			//如果找到我们需要的name类,直接返回当前clazz
            if (clazz != null) {
                return clazz;
            }
        }
 
        if (dexElementsSuppressedExceptions != null) {
            suppressed.addAll(Arrays.asList(dexElementsSuppressedExceptions));
        }
        return null;
    }
}

看注释2处 此时会通过makeDexElements方法生成一个Element数组,紧接着当前类中的findClass方法又会调用DexPathList中的Element类的findClass方法。


static class Element {
      
        private final File path;
 
        private final DexFile dexFile;
 
        private ClassPathURLStreamHandler urlHandler;
        private boolean initialized;
 
		...
		
      
		...
		
        public Class<?> findClass(String name, ClassLoader definingContext,List<Throwable> suppressed) {
			// 3 通过loadClassBinaryName方法寻找name类,找到即返回它,否则返回null 
            return dexFile != null ? dexFile.loadClassBinaryName(name, definingContext, suppressed)
                    : null;
        }
 
    }

注释3 处   通过loadClassBinaryName 最后调用native层 defineClassNative的方法 分析到这里可以看出,最终进行Class字节码的加载操作,是通过底层的native方法来完成的。

到此这篇关于Java源码解析之ClassLoader的文章就介绍到这了,更多相关Java ClassLoader内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯