Android開發藝術探索 第13章綜合技術(檢測Crash,65k問題,動態加載)&& 第14章 JNI與NDK編程 讀書筆記

本章主要講解1. CrashHandler來監視App的crash信息;2. Android65536問題;3. Android動態加載;4. 反編譯。


13.1 使用CrashHandler來獲取應用的crash信息

如何檢測崩潰并了解詳細的crash信息? 首先需實現一個uncaughtExceptionHandler對象,在它的uncaughtException方法中獲取異常信息并將其存儲到SD卡或者上傳到服務器中,然后調用Thread的setDefaultUncaughtExceptionHandler為當前進程的所有線程設置異常處理器。

public class CrashHandler implements Thread.UncaughtExceptionHandler {
    private static final String TAG = "CrashHandler";
    private static final boolean DEBUG = true;

    private static final String PATH = Environment.getExternalStorageDirectory().getPath() + "/CrashTest/log/";
    private static final String FILE_NAME = "crash";
    private static final String FILE_NAME_SUFFIX = ".trace";

    private static CrashHandler sInstance = new CrashHandler();
    private Thread.UncaughtExceptionHandler mDefaultCrashHandler;
    private Context mContext;

    private CrashHandler() {
    }

    public static CrashHandler getInstance() {
        return sInstance;
    }

    public void init(Context context) {
        mDefaultCrashHandler = Thread.getDefaultUncaughtExceptionHandler();
        Thread.setDefaultUncaughtExceptionHandler(this);
        mContext = context.getApplicationContext();
    }

    /**
     * 這個是最關鍵的函數,當程序中有未被捕獲的異常,系統將會自動調用#uncaughtException方法
     * thread為出現未捕獲異常的線程,ex為未捕獲的異常,有了這個ex,我們就可以得到異常信息。
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        try {
            //導出異常信息到SD卡中
            dumpExceptionToSDCard(ex);
            uploadExceptionToServer();
            //這里可以通過網絡上傳異常信息到服務器,便于開發人員分析日志從而解決bug
        } catch (IOException e) {
            e.printStackTrace();
        }

        ex.printStackTrace();

        //如果系統提供了默認的異常處理器,則交給系統去結束我們的程序,否則就由我們自己結束自己
        if (mDefaultCrashHandler != null) {
            mDefaultCrashHandler.uncaughtException(thread, ex);
        } else {
            Process.killProcess(Process.myPid());
        }
    }

    private void dumpExceptionToSDCard(Throwable ex) throws IOException {
        //偽代碼 本方法用于實現將錯誤信息存儲到SD卡中
    }

  

    private void uploadExceptionToServer() {
        //偽代碼 本方法用于將錯誤信息上傳至服務器
    }
}

然后在Application初始化的時候為線程設置CrashHandler,這樣之后,Crash就會通過我們自己的異常處理器來處理異常了。

public class BaseApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        CrashHandler crashHandler = CrashHandler.getInstance();
        crashHandler.init(this);
    } 
}

想看書中完整源碼請點擊


13.2 使用multidex來解決方法數越界

  1. 在Android中單個dex文件所能包含的最大方法數為65536,這個是包含Android FrameWork、依賴jar包以及應用本身代碼中的所有方法。達到這個65536后,編譯器編譯時會拋出DexIndexOverflowException異常。
  2. 如何解決?Google提供了multidex解決方案。在Android5.0之前需要引入Google提供的android-support-multidex.jar;從5.0開始系統默認支持了multidex,它可以從apk文件中加載多個dex文件。
  3. 使用步驟
  4. 修改對應工程目錄下的build.gradle文件,在defaultConfig中添加multiDexEnabled true這個配置項。
  5. 在build.gradle的dependencies中添加multidex的依賴:compile 'com.android.support:multidex:1.0.0'
android {
    compileSdkVersion 23
    buildToolsVersion "23.0.3"
    defaultConfig {
        applicationId "cn.hudp.androiddevartnote"
        minSdkVersion 14
        targetSdkVersion 23
        versionCode 1
        versionName "1.0"
        multiDexEnabled true  //關鍵部分
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}
dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:23.4.0'
    compile 'com.android.support:multidex:1.0.0' //關鍵部分
}
  1. 代碼中加入支持multidex功能。
    第一種方案,在manifest文件中指定Application為MultiDexApplication。
    第二種方案,讓應用的Application繼承MultiDexApplication。
    第三種方案,重寫attachBaseContext方法,這個方法比onCreate還要先執行。
public class BaseApplication extends Application {
  @Override
  protected void attachBaseContext(Context base) {
        super.attachBaseContext(base);
        MultiDex.install(this);
    }
}

采用上面的配置項后,如果這個應用方法數沒有越界,那么Gradle是不會生成多個dex文件的,當方法數越界后,Gradle就會在apk中打包2個或多個dex文件。當需要指定主dex文件中所包含的類,這時候就需要通過--multi-dex-list來選項來實現這個功能。

//在對應工程目錄下的build.gradle文件,加入
afterEvaluate {
    println "afterEvaluate"
    tasks.matching {
        it.name.startsWith('dex')
    }.each { dx ->
        def listFile = project.rootDir.absolutePath + '/app/maindexlist.txt'
        println "root dir:" + project.rootDir.absolutePath
        println "dex task found: " + dx.name
        if (dx.additionalParameters == null) {
            dx.additionalParameters = []
        }
        dx.additionalParameters += '--multi-dex'
        dx.additionalParameters += '--main-dex-list=' + listFile
        dx.additionalParameters += '--minimal-main-dex'
    }
} 
//maindexlist.txt
com/ryg/multidextest/TestApplication.class
com/ryg/multidextest/MainActivity.class
// multidex 這9個類必須在主Dex中
android/support/multidex/MultiDex.class
android/support/multidex/MultiDexApplication.class
android/support/multidex/MultiDexExtractor.class
android/support/multidex/MultiDexExtractor$1.class
android/support/multidex/MultiDex$V4.class
android/support/multidex/MultiDex$V14.class
android/support/multidex/MultiDex$V19.class
android/support/multidex/ZipUtil.class
android/support/multidex/ZipUtil$CentralDirectory.class

需要注意multidex的jar中的9個類必須要打包到主dex中,因為Application的attachBaseContext方法中需要用到MultiDex.install(this)需要用到MultiDex。

Multidex的缺點:

  1. 啟動速度會降低,由于應用啟動時會加載額外的dex文件,這將導致應用的啟動速度降低,甚至產生ANR現象。
  2. 因為Dalvik linearAlloc的bug,可以導致使用multidex的應用無法在Android4.0之前的手機上運行,需要做大量兼容性測試。

13.3 Android動態加載技術

各種插件化方案都需要解決3個基礎性問題:

  1. 資源訪問,因為插件中凡是以R開頭的資源文件都不能訪問了。
  2. Activity的生命周期管理,因為宿主動態將Activity.java加載到內存的時候,是不具備Activity的任何特性的,只是一個普通的java類。
  3. ClassLoader的管理,為了避免多個ClassLoader加載了同一個類所引發的類型轉換錯誤。

介紹兩個具有代表性的插件化解決方案

  1. dynamic-load-apk
  2. Droid Plugin
    DL方案對Framework的表層做了處理,依賴that語法,編寫插件代碼和主程序代碼需單獨區分;而DroidPlugin通過Hook增強了Framework層的很多系統服務,開發插件就跟開發獨立app差不多;就拿Activity生命周期的管理來說,DL的代理方式就像是牽線木偶,插件是操縱傀儡;而DroidPlugin則是借尸還魂,插件是有血有肉的系統管理的真正組件;DroidPlugin Hook了系統幾乎所有的Sevice,欺騙了大部分的系統API;
    (PS:這部分DL以及DP的分析來自于weishu

13.4 反編譯初步

  1. 使用dex2jar和jd-gui反編譯apk
  2. 使用apktool對apk進行二次打包
    這塊不是難點,網上資料也相當多,跳過啦。

第十四章 JNI與NDK編程

參考博主的另一篇文章Android NDK初體驗,帶你快速進入NDK的世界

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,119評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,382評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 176,038評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,853評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,616評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,112評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,192評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,355評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,869評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,727評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,928評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,467評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,165評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,570評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,813評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,585評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,892評論 2 372

推薦閱讀更多精彩內容