日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Java源碼解析之ClassLoader

瀏覽:162日期:2022-08-12 09:16:48
目錄一、前言二、java 中的 ClassLoader三、Android 中的 ClassLoader四、雙親委派機制五、源碼分析一、前言

一個完整的Java應(yīng)用程序,當程序在運行時,即會調(diào)用該程序的一個入口函數(shù)來調(diào)用系統(tǒng)的相關(guān)功能,而這些功能都被封裝在不同的class文件當中,所以經(jīng)常要從這個class文件中要調(diào)用另外一個class文件中的方法,如果另外一個文件不存在的,則會引發(fā)系統(tǒng)異常。而程序在啟動的時候,并不會一次性加載程序所要用的所有class文件,而是根據(jù)程序的需要,通過Java的類加載機制(ClassLoader)來動態(tài)加載某個class文件到內(nèi)存當中的,從而只有class文件被載入到了內(nèi)存之后,才能被其它class所引用。所以ClassLoader就是用來動態(tài)加載class文件到內(nèi)存當中用的。

Android平臺上虛擬機運行的是Dex字節(jié)碼,一種對class文件優(yōu)化的產(chǎn)物,傳統(tǒng)Class文件是一個Java源碼文件會生成一個.class文件,而Android是把所有Class文件進行合并,優(yōu)化,然后生成一個最終的class.dex,目的是把不同class文件重復的東西只需保留一份,如果我們的Android應(yīng)用不進行分dex處理,最后一個應(yīng)用的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系統(tǒng)啟動時會使用BootClassLoader來預加載常用類,與Java中的Bootstrap ClassLoader不同的是,它并不是由C/C++代碼實現(xiàn),而是由Java實現(xiàn)的。BootClassLoader是ClassLoader的一個內(nèi)部類。PathClassLoader

負責加載已經(jīng)安裝的Apk,也就是/data/app/package 下的apk文件,也可以加載/vendor/lib, /system/lib下的nativeLibrary

DexClassLoader

負責加載可以加載一個未安裝的apk文件。

四、雙親委派機制

每一個 ClassLoader 中都有一個 parent 對象,代表的是父類加載器,在加載一個類的時候,會先使用父類加載器去加載,如果在父類加載器中沒有找到,自己再進行加載,如果 parent 為空,那么就用系統(tǒng)類加載器來加載。通過這樣的機制可以保證系統(tǒng)類都是由系統(tǒng)類加載器加載的。 下面是 ClassLoader 的 loadClass 方法的具體實現(xiàn)。

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.現(xiàn)在我們看下 BaseDexClassLoader 繼承自ClassLoader

public class BaseDexClassLoader extends ClassLoader{ ...//存放需要加載的dexListprivate final DexPathList pathList;/** * * @param dexPath 需要加載的dex文件所在的路徑 * @param optimizedDirectory Android系統(tǒng)將dex文件進行優(yōu)化后所生成的ODEX文件的存放路徑,該路徑必須是一個內(nèi)部存儲路徑。 * @param librarySearchPath 目標類所使用的c、c++庫存放的路徑 * @param parent 該加載器的父加載器,一般為當前執(zhí)行類的加載器 */ public BaseDexClassLoader(String dexPath, File optimizedDirectory, String librarySearchPath, ClassLoader parent) {this(dexPath, optimizedDirectory, librarySearchPath, parent, false); } /** * * @param dexPath * @param optimizedDirectory * @param librarySearchPath * @param parent * @param isTrusted 是否已信任,關(guān)系到是否可調(diào)用隱藏API */ public BaseDexClassLoader(String dexPath, File optimizedDirectory, String librarySearchPath, ClassLoader parent, boolean isTrusted) {super(parent);this.pathList = new DexPathList(this, dexPath, librarySearchPath, null, isTrusted); ... } /** * * @param dexFiles 字節(jié)緩存數(shù)組的dex文件 * @param parent 該加載器的父加載器 */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); }/** * 通過完整的類名尋找對應(yīng)的類 * @param name 傳入一個完整的類名 * @return * @throws ClassNotFoundException */@Override protected Class<?> findClass(String name) throws ClassNotFoundException {List<Throwable> suppressedExceptions = new ArrayList<Throwable>();//1 在pathList中尋找name對應(yīng)的類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 { /** * * @param dexPath dex文件路徑集合 * @param parent 父加載器 */ public PathClassLoader(String dexPath, ClassLoader parent) {//調(diào)用父類BaseDexClassLoader 四參構(gòu)造方法super(dexPath, null, null, parent); } /** * * @param dexPath dex文件路徑集合 * @param librarySearchPath 包含 C/C++庫的路徑集合,多個路徑用文件分隔符分隔分割,可以為null * @param parent 父加載器 */ public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {//調(diào)用父類BaseDexClassLoader 四參構(gòu)造方法super(dexPath, null, librarySearchPath, parent); }}

public class DexClassLoader extends BaseDexClassLoader { /** * * @param dexPath dex文件路徑集合,多個路徑用文件分隔符分隔,默認文件分隔符為':' * @param optimizedDirectory 解壓的dex文件存儲路徑,這個路徑必須是一個內(nèi)部存儲路徑,一般情況下使用當前應(yīng)用程序的私有路徑 * @param librarySearchPath 包含 C/C++ 庫的路徑集合,多個路徑用文件分隔符分隔分割,可以為null * @param parent 父加載器 */ public DexClassLoader(String dexPath, String optimizedDirectory, String librarySearchPath, ClassLoader parent) {// 調(diào)用父類BaseDexClassLoader 四參構(gòu)造方法,在API26以上,librarySearchPath參數(shù)已棄用,使用此方法super(dexPath, null, librarySearchPath, parent);// API26及以下使用此方法// super(dexPath, new File(optimizedDirectory), librarySearchPath, parent); }}

我們看1 處 pathList 的 findClass 是如何查找的

/*package*/ final class DexPathList { private static final String DEX_SUFFIX = '.dex'; private static final String zipSeparator = '!/'; private final ClassLoader definingContext; // dex/resource 存放dex的數(shù)組 private Element[] dexElements; // 存放本地庫文件的列表 private final List<File> nativeLibraryDirectories; // 存放系統(tǒng)本地庫文件的列表 private final List<File> systemNativeLibraryDirectories; // 存放創(chuàng)建dexElement列表時引發(fā)異常的列表 private IOException[] dexElementsSuppressedExceptions;DexPathList(ClassLoader definingContext, String dexPath, String librarySearchPath, File optimizedDirectory, boolean isTrusted) {... this.definingContext = definingContext; //BaseDexClassLoader構(gòu)造器中會傳入其本身 ArrayList<IOException> suppressedExceptions = new ArrayList<IOException>();// 通過dexPath路徑使用分隔符將其轉(zhuǎn)換成dexElements列表this.dexElements = makeDexElements(splitDexPath(dexPath), optimizedDirectory, suppressedExceptions, definingContext, isTrusted);... } // 為本地庫搜索路徑生成一個directory/zip path元素數(shù)組 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 文件保存,注意第二個參數(shù)傳的底 null elements[elementsPos++] = new Element(dex, null); } } catch (IOException suppressed) { System.logE('Unable to load dex file: ' + file, suppressed); suppressedExceptions.add(suppressed); } } else { //如果不是目錄且不是 .dex 結(jié)尾的,那么他可能是 jar。加載這個文件 dex = loadDexFile(file, optimizedDirectory, loader, elements); if (dex == null) { elements[elementsPos++] = new Element(file); } else { // 保存dex 文件 和 當前的 file,這種情況下可能是 jar,具體可以看這個構(gòu)造方法 elements[elementsPos++] = new Element(dex, file); } if (dex != null && isTrusted) { //如果dex對象不為空且是允許信任狀態(tài)dex.setTrusted(); // 將此dex對象設(shè)置為已信任,它可以訪問平臺的隱藏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數(shù)組,緊接著當前類中的findClass方法又會調(diào)用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 最后調(diào)用native層 defineClassNative的方法 分析到這里可以看出,最終進行Class字節(jié)碼的加載操作,是通過底層的native方法來完成的。

到此這篇關(guān)于Java源碼解析之ClassLoader的文章就介紹到這了,更多相關(guān)Java ClassLoader內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Java
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产色综合网| 中文在线中文资源| 国产乱码精品一区二区三区四区| 久久久精品日韩| 日韩二区在线观看| 日韩国产专区| 精品在线91| 在线亚洲欧美| 成人av二区| 国产精品美女久久久| 在线精品视频在线观看高清| 四虎影视精品| 欧美综合另类| 国产一区导航| 丝袜诱惑制服诱惑色一区在线观看| 国内精品99| 欧美va亚洲va日韩∨a综合色| 日韩亚洲一区在线| 亚洲精品888| 久久视频一区| 在线 亚洲欧美在线综合一区| 不卡在线一区二区| 亚洲有吗中文字幕| 国产亚洲欧美日韩精品一区二区三区 | 亚洲人成网77777色在线播放| 伊人成人在线视频| 欧美日韩视频| 亚洲免费成人av在线| 97国产成人高清在线观看| 91精品精品| 国产精品美女久久久浪潮软件| 亚洲+小说+欧美+激情+另类| 国产伦精品一区二区三区千人斩| 国产福利一区二区三区在线播放| 另类综合日韩欧美亚洲| 国产一区二区三区网| 亚洲欧美综合| 欧美视频精品全部免费观看| 国产黄大片在线观看| 日韩在线观看不卡| 美女视频黄免费的久久| 久久免费高清| 国产精品中文字幕制服诱惑| 99精品视频在线| 国产精久久久| 亚洲欧美视频| 国产中文在线播放| 91伊人久久| 激情婷婷久久| 国产激情欧美| 久久亚洲欧洲| av中文字幕在线观看第一页| 日韩av中文在线观看| 丝袜av一区| 精品国产一级| 日韩欧美三区| 国产黄大片在线观看| 日本成人在线网站| 欧美日韩国产在线观看网站 | 久久国产直播| 国产成人精品福利| 国产欧美大片| 日本成人在线不卡视频| 九九在线精品| av一区在线| 欧美日韩在线观看首页| 日韩高清电影一区| 国产亚洲欧美日韩在线观看一区二区| 国产精品免费不| 日韩激情av在线| 五月精品视频| 欧美日韩激情| 综合激情一区| 蜜臀久久99精品久久久久久9| 激情久久99| 国产伦精品一区二区三区视频| 四虎精品一区二区免费| 日韩精品福利一区二区三区| 99热免费精品| 中文精品电影| 亚洲日本欧美| 99精品99| 中文精品视频| 蜜臀av性久久久久蜜臀aⅴ流畅| 影音先锋久久| 久久影视三级福利片| 精品一区91| 日韩欧美精品| 亚洲国产一区二区在线观看| 午夜一级在线看亚洲| 亚洲区欧美区| 久久av影视| 成人美女视频| 国产亚洲福利| 日韩一区二区三区高清在线观看| 欧美日韩亚洲一区三区| 欧美精品导航| 99精品国产一区二区三区| 日韩中文欧美在线| 国产精品久久久久久模特| 久久精品天堂| 亚洲免费精品| 久久99久久久精品欧美| 欧美一级精品| 日本少妇精品亚洲第一区| 在线中文字幕播放| 天堂成人免费av电影一区| 国产日本精品| 激情久久中文字幕| 国产精品麻豆成人av电影艾秋| 日韩理论片av| 蜜臀av性久久久久蜜臀aⅴ四虎| 国产精品成人自拍| 99在线|亚洲一区二区| 国产精品调教| 水野朝阳av一区二区三区| 大香伊人久久精品一区二区| 视频一区二区国产| 日本不良网站在线观看| 亚久久调教视频| 精品视频高潮| 天堂va欧美ⅴa亚洲va一国产| 日韩影院二区| 欧美日韩调教| 亚洲免费影视| 日韩免费视频| 日本亚洲最大的色成网站www | 麻豆免费精品视频| 国产精品老牛| 精品在线99| 国产福利资源一区| 日本不卡视频在线| 亚洲日本久久| 樱桃成人精品视频在线播放| 日韩国产专区| 国产色播av在线| 麻豆一区二区在线| 麻豆91精品视频| 国产精品观看| 国产精久久久| 日韩不卡一区二区| 中文在线日韩| 亚洲精品日本| 日韩国产欧美三级| 日本亚洲欧洲无免费码在线| 蜜臀va亚洲va欧美va天堂 | 丁香六月综合| 成午夜精品一区二区三区软件| 狠狠干综合网| 国产一区视频在线观看免费| 免费在线成人| 日韩国产欧美| 日韩欧美精品一区| 成人污污视频| 在线看片国产福利你懂的| 精品国产一级| 国产一区二区三区四区五区| 99久久夜色精品国产亚洲1000部| 免费观看久久久4p| 国产粉嫩在线观看| 日韩精品免费视频人成| 日本不良网站在线观看| 最新国产精品| 亚洲a在线视频| 久久国产乱子精品免费女| 久久高清免费| 精品久久美女| 日韩高清不卡在线| 国产精选一区| 91成人超碰| 日本午夜精品一区二区三区电影| 伊人网在线播放| 国产精品三p一区二区| 首页国产欧美日韩丝袜| 日韩一区二区中文| 欧美激情三区| 日韩欧美在线精品| 日韩午夜av在线| 美女一区网站| 福利精品在线| 国产日韩欧美中文在线| 亚洲一区二区日韩| 免费中文字幕日韩欧美| 国产亚洲一区二区手机在线观看| 国产欧美日韩一区二区三区四区| 免费不卡在线观看| 免费中文字幕日韩欧美| 欧美午夜不卡| 五月综合激情| 国产在线不卡| av一区二区高清| 欧美亚洲精品在线| 欧美日韩国产传媒| 亚洲欧美日韩一区在线观看| 日韩午夜av在线| 视频一区国产视频| 偷拍亚洲精品| 国产精品一区二区三区av| 青青草伊人久久| 欧美日本精品|