分析google的multidex庫

来源:http://www.cnblogs.com/daohen/archive/2016/02/27/5223233.html
-Advertisement-
Play Games

我們在開發項目時,喜歡引入好多的第三方包,大大的方便了我們的開發,但同時,因為android方法總數的限制,不能超過65k,然而呢,隨著我們的開發,65k最終還是會超過,所以,google就給出了這個解決方案,但一直好奇它是內部是怎麼實現的,我們今天就來根據源碼來看看這個包到底做了什麼,怎麼把多個d


  我們在開發項目時,喜歡引入好多的第三方包,大大的方便了我們的開發,但同時,因為android方法總數的限制,不能超過65k,然而呢,隨著我們的開發,65k最終還是會超過,所以,google就給出了這個解決方案,但一直好奇它是內部是怎麼實現的,我們今天就來根據源碼來看看這個包到底做了什麼,怎麼把多個dex讀取出來的。先看下這個包裡面都有哪些類:

                

  我們首先看MultiDexApplication,只要我們我們自己的Application繼承MultiDexApplication就可以解決問題,那我們就來看看,它裡面做了什麼,只有一個方法,重寫了attachBaseContext()。

1 protected void attachBaseContext(Context base) {
2     super.attachBaseContext(base);
3     MultiDex.install(this);
4 }

  我們看到,具體的實現是直接調用的Multidex,隨後我們來看下install()裡面有些什麼(關鍵代碼):

 1     public static void install(Context context) {
 2         try {
 3               .
 4               .
 5               .
 6                 /* The patched class loader is expected to be a descendant of
 7                  * dalvik.system.BaseDexClassLoader. We modify its
 8                  * dalvik.system.DexPathList pathList field to append additional DEX
 9                  * file entries.
10                  */
11                 ClassLoader loader;
12                 try {
13                     loader = context.getClassLoader();
14                 } catch (RuntimeException e) {
15                 }
16                 
17                 File dexDir = new File(applicationInfo.dataDir, SECONDARY_FOLDER_NAME);
18                 List<File> files = MultiDexExtractor.load(context, applicationInfo, dexDir, false);
19                 if (checkValidZipFiles(files)) {
20                     installSecondaryDexes(loader, dexDir, files);
21                 } else {
22                   .
23                   .
24                   .
25                 }
26             }
27         } catch (Exception e) {
28         }
29     }

  通過ClassLoader的註釋我們已經知道,主dex文件的路徑被存儲在BaseDexClassLoader中的pathList,這樣就清楚了,下麵所要做的就是把其它的dex文件路徑也找出來,添加到pathList上面即可。這裡面的loader就是BaseDexClassLoader的實例

  行18 MultiDexExtractor這個類,從字面上即可知道它是提取dex信息的,load方法即會根據dex命名規則從指定路徑下提取其它dex文件(不包含主dex),具體實現可自行看實現,這樣,所有的次dex文件都被提取出來,賦值給了files

  行19 會檢查所有以上文件是否是有效的zip文件,如果有一個false,就會重新提取dex文件

  行20 檢查沒有問題後,就會執行installSecondaryDexes(),已經很明顯知道接下來要做什麼了,我們來看這個方法裡面都做了什麼。

 1     private static void installSecondaryDexes(ClassLoader loader, File dexDir, List<File> files)
 2             throws IllegalArgumentException, IllegalAccessException, NoSuchFieldException,
 3             InvocationTargetException, NoSuchMethodException, IOException {
 4         if (!files.isEmpty()) {
 5             if (Build.VERSION.SDK_INT >= 19) {
 6                 V19.install(loader, files, dexDir);
 7             } else if (Build.VERSION.SDK_INT >= 14) {
 8                 V14.install(loader, files, dexDir);
 9             } else {
10                 V4.install(loader, files);
11             }
12         }
13     }

  代碼寫的很明顯,我們來單看V19的實現 

 

 1         private static void install(ClassLoader loader, List<File> additionalClassPathEntries,
 2                 File optimizedDirectory)
 3                         throws IllegalArgumentException, IllegalAccessException,
 4                         NoSuchFieldException, InvocationTargetException, NoSuchMethodException {
 5             /* The patched class loader is expected to be a descendant of
 6              * dalvik.system.BaseDexClassLoader. We modify its
 7              * dalvik.system.DexPathList pathList field to append additional DEX
 8              * file entries.
 9              */
10             Field pathListField = findField(loader, "pathList");
11             Object dexPathList = pathListField.get(loader);
12             ArrayList<IOException> suppressedExceptions = new ArrayList<IOException>();
13             expandFieldArray(dexPathList, "dexElements", makeDexElements(dexPathList,
14                     new ArrayList<File>(additionalClassPathEntries), optimizedDirectory,
15                     suppressedExceptions));
16             .
17             .
18             .
19         }

 

 

 

  我們前面說的註釋又出現了,看來,具體的實現操作就在這裡了,主要是用反射來修改它的值,我們主要來看幾個主要的操作方法:

  行13 這裡面有兩個方法需要註意makeDexElements()和expandFieldArray(),先來說第一個,我們知道在實例化BaseDexClassLoader時,會把主dex的路徑信息存放到pathList裡面,而DexPathList內部其它是把dex的路徑存儲在了一個Element數組中,所以,看makeDexElements()就知道,這個方法,會把傳入的dex文件通過反射組裝成適合DexPathList內部用的Element數組。

 1         private static Object[] makeDexElements(
 2                 Object dexPathList, ArrayList<File> files, File optimizedDirectory,
 3                 ArrayList<IOException> suppressedExceptions)
 4                         throws IllegalAccessException, InvocationTargetException,
 5                         NoSuchMethodException {
 6             Method makeDexElements =
 7                     findMethod(dexPathList, "makeDexElements", ArrayList.class, File.class,
 8                             ArrayList.class);
 9 
10             return (Object[]) makeDexElements.invoke(dexPathList, files, optimizedDirectory,
11                     suppressedExceptions);
12         }

  看它代碼是反射調用DexPathList裡面的makeDexElements(),但我一直沒有找到這個方法,不知道是代碼版本的問題還是其它原因,望知道的大神告訴一下。接下來我們再看expandFieldArray()

    private static void expandFieldArray(Object instance, String fieldName,
            Object[] extraElements) throws NoSuchFieldException, IllegalArgumentException,
            IllegalAccessException {
        Field jlrField = findField(instance, fieldName);
        Object[] original = (Object[]) jlrField.get(instance);
        Object[] combined = (Object[]) Array.newInstance(
                original.getClass().getComponentType(), original.length + extraElements.length);
        System.arraycopy(original, 0, combined, 0, original.length);
        System.arraycopy(extraElements, 0, combined, original.length, extraElements.length);
        jlrField.set(instance, combined);
    }

  可看出,又重新建立了一個Element[],把原來和新的數據都放上去,最後賦給dexElements。這樣就可以把所有的dex都讀取出來了。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • RecyclerView是加強版的ListView,用於在有限的視窗中展示大量的數據,而LoadMoreRecyclerView則是為RecyclerView增加了載入更多的功能,先來看效果: 三種載入方式: 自動載入: 手動載入: 使用圖片動畫載入: 項目簡介: demo中的下拉刷新採用的是Swi
  • AndroidManifest.xml <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.y
  • AndroidManifest.xml <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.y
  • 本文簡單介紹了Android(Linux)串口驅動線路規程的使用。
  • 分類:C#、Android、VS2015; 創建日期:2016-02-28 一、簡介 廣播(Broadcast):其功能類似於收音機的廣播,你只要調到那個台(只要在接收的類中註冊了要接收的廣播),就能收到這個台播放的信息。 通知(Notifications):安卓的服務組件(Android Serv
  • 在iOS開發過程中,我一直習慣於使用C語法里的基本類型,而很少用(除非必須使用)Foundation的數據類型。最近看了一些資料,發現自己這樣寫可能有風險,雖然目前沒遇到過相關的問題,但這是非常需要註意的一點。 新博客 "wossoneri.com" 壞習慣的開端 初寫iOS時,我做的是把原項目從
  • 安卓第四天筆記-Sqlite 1.資料庫的創建運行與更新 1.1.創建一個類繼承SqliteOpenHelper 1.2.創建構造方法 /** * 資料庫創建類 * @author 劉楠 * * 2016-2-20上午10:04:34 */ public class DbSqliteOpenHelp
  • 分類:C#、Android、VS2015; 創建日期:2016-02-27 一、簡介 Android使用的文件系統是基於Linux的文件系統,在Android應用程式中,開發人員既可以建立和訪問程式自身的私有文件,也可以訪問保存在資源目錄中的原始文件和XML文件,同時還可以將文件保存在SD卡等外部存...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...