Android開發(fā)藝術(shù)探索 第13章綜合技術(shù)(檢測(cè)Crash龙亲,65k問題,動(dòng)態(tài)加載)&& 第14章 JNI與NDK編程 讀書筆記

本章主要講解1. CrashHandler來監(jiān)視App的crash信息悍抑;2. Android65536問題鳄炉;3. Android動(dòng)態(tài)加載;4. 反編譯搜骡。


13.1 使用CrashHandler來獲取應(yīng)用的crash信息

如何檢測(cè)崩潰并了解詳細(xì)的crash信息拂盯? 首先需實(shí)現(xiàn)一個(gè)uncaughtExceptionHandler對(duì)象,在它的uncaughtException方法中獲取異常信息并將其存儲(chǔ)到SD卡或者上傳到服務(wù)器中记靡,然后調(diào)用Thread的setDefaultUncaughtExceptionHandler為當(dāng)前進(jìn)程的所有線程設(shè)置異常處理器谈竿。

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();
    }

    /**
     * 這個(gè)是最關(guān)鍵的函數(shù),當(dāng)程序中有未被捕獲的異常摸吠,系統(tǒng)將會(huì)自動(dòng)調(diào)用#uncaughtException方法
     * thread為出現(xiàn)未捕獲異常的線程空凸,ex為未捕獲的異常,有了這個(gè)ex寸痢,我們就可以得到異常信息呀洲。
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        try {
            //導(dǎo)出異常信息到SD卡中
            dumpExceptionToSDCard(ex);
            uploadExceptionToServer();
            //這里可以通過網(wǎng)絡(luò)上傳異常信息到服務(wù)器,便于開發(fā)人員分析日志從而解決bug
        } catch (IOException e) {
            e.printStackTrace();
        }

        ex.printStackTrace();

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

    private void dumpExceptionToSDCard(Throwable ex) throws IOException {
        //偽代碼 本方法用于實(shí)現(xiàn)將錯(cuò)誤信息存儲(chǔ)到SD卡中
    }

  

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

然后在Application初始化的時(shí)候?yàn)榫€程設(shè)置CrashHandler,這樣之后族壳,Crash就會(huì)通過我們自己的異常處理器來處理異常了憔辫。

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

想看書中完整源碼請(qǐng)點(diǎn)擊


13.2 使用multidex來解決方法數(shù)越界

  1. 在Android中單個(gè)dex文件所能包含的最大方法數(shù)為65536,這個(gè)是包含Android FrameWork仿荆、依賴jar包以及應(yīng)用本身代碼中的所有方法贰您。達(dá)到這個(gè)65536后,編譯器編譯時(shí)會(huì)拋出DexIndexOverflowException異常拢操。
  2. 如何解決锦亦?Google提供了multidex解決方案。在Android5.0之前需要引入Google提供的android-support-multidex.jar令境;從5.0開始系統(tǒng)默認(rèn)支持了multidex杠园,它可以從apk文件中加載多個(gè)dex文件。
  3. 使用步驟
  4. 修改對(duì)應(yīng)工程目錄下的build.gradle文件舔庶,在defaultConfig中添加multiDexEnabled true這個(gè)配置項(xiàng)抛蚁。
  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  //關(guān)鍵部分
    }
    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' //關(guān)鍵部分
}
  1. 代碼中加入支持multidex功能陈醒。
    第一種方案,在manifest文件中指定Application為MultiDexApplication瞧甩。
    第二種方案钉跷,讓應(yīng)用的Application繼承MultiDexApplication。
    第三種方案肚逸,重寫attachBaseContext方法爷辙,這個(gè)方法比onCreate還要先執(zhí)行。
public class BaseApplication extends Application {
  @Override
  protected void attachBaseContext(Context base) {
        super.attachBaseContext(base);
        MultiDex.install(this);
    }
}

采用上面的配置項(xiàng)后朦促,如果這個(gè)應(yīng)用方法數(shù)沒有越界膝晾,那么Gradle是不會(huì)生成多個(gè)dex文件的,當(dāng)方法數(shù)越界后务冕,Gradle就會(huì)在apk中打包2個(gè)或多個(gè)dex文件血当。當(dāng)需要指定主dex文件中所包含的類,這時(shí)候就需要通過--multi-dex-list來選項(xiàng)來實(shí)現(xiàn)這個(gè)功能禀忆。

//在對(duì)應(yīng)工程目錄下的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個(gè)類必須在主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個(gè)類必須要打包到主dex中,因?yàn)锳pplication的attachBaseContext方法中需要用到MultiDex.install(this)需要用到MultiDex油湖。

Multidex的缺點(diǎn):

  1. 啟動(dòng)速度會(huì)降低,由于應(yīng)用啟動(dòng)時(shí)會(huì)加載額外的dex文件领跛,這將導(dǎo)致應(yīng)用的啟動(dòng)速度降低乏德,甚至產(chǎn)生ANR現(xiàn)象。
  2. 因?yàn)镈alvik linearAlloc的bug吠昭,可以導(dǎo)致使用multidex的應(yīng)用無法在Android4.0之前的手機(jī)上運(yùn)行喊括,需要做大量兼容性測(cè)試。

13.3 Android動(dòng)態(tài)加載技術(shù)

各種插件化方案都需要解決3個(gè)基礎(chǔ)性問題:

  1. 資源訪問矢棚,因?yàn)椴寮蟹彩且訰開頭的資源文件都不能訪問了郑什。
  2. Activity的生命周期管理,因?yàn)樗拗鲃?dòng)態(tài)將Activity.java加載到內(nèi)存的時(shí)候蒲肋,是不具備Activity的任何特性的蘑拯,只是一個(gè)普通的java類。
  3. ClassLoader的管理兜粘,為了避免多個(gè)ClassLoader加載了同一個(gè)類所引發(fā)的類型轉(zhuǎn)換錯(cuò)誤申窘。

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

  1. dynamic-load-apk
  2. Droid Plugin
    DL方案對(duì)Framework的表層做了處理,依賴that語法孔轴,編寫插件代碼和主程序代碼需單獨(dú)區(qū)分剃法;而DroidPlugin通過Hook增強(qiáng)了Framework層的很多系統(tǒng)服務(wù),開發(fā)插件就跟開發(fā)獨(dú)立app差不多路鹰;就拿Activity生命周期的管理來說贷洲,DL的代理方式就像是牽線木偶收厨,插件是操縱傀儡;而DroidPlugin則是借尸還魂优构,插件是有血有肉的系統(tǒng)管理的真正組件诵叁;DroidPlugin Hook了系統(tǒng)幾乎所有的Sevice,欺騙了大部分的系統(tǒng)API俩块;
    (PS:這部分DL以及DP的分析來自于weishu

13.4 反編譯初步

  1. 使用dex2jar和jd-gui反編譯apk
  2. 使用apktool對(duì)apk進(jìn)行二次打包
    這塊不是難點(diǎn)黎休,網(wǎng)上資料也相當(dāng)多,跳過啦玉凯。

第十四章 JNI與NDK編程

參考博主的另一篇文章Android NDK初體驗(yàn)势腮,帶你快速進(jìn)入NDK的世界

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市漫仆,隨后出現(xiàn)的幾起案子捎拯,更是在濱河造成了極大的恐慌,老刑警劉巖盲厌,帶你破解...
    沈念sama閱讀 219,366評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件署照,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡吗浩,警方通過查閱死者的電腦和手機(jī)建芙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,521評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來懂扼,“玉大人禁荸,你說我怎么就攤上這事》” “怎么了赶熟?”我有些...
    開封第一講書人閱讀 165,689評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)陷嘴。 經(jīng)常有香客問我映砖,道長(zhǎng),這世上最難降的妖魔是什么灾挨? 我笑而不...
    開封第一講書人閱讀 58,925評(píng)論 1 295
  • 正文 為了忘掉前任邑退,我火速辦了婚禮,結(jié)果婚禮上劳澄,老公的妹妹穿的比我還像新娘瓜饥。我一直安慰自己,他們只是感情好浴骂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,942評(píng)論 6 392
  • 文/花漫 我一把揭開白布乓土。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪趣苏。 梳的紋絲不亂的頭發(fā)上狡相,一...
    開封第一講書人閱讀 51,727評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音食磕,去河邊找鬼尽棕。 笑死,一個(gè)胖子當(dāng)著我的面吹牛彬伦,可吹牛的內(nèi)容都是我干的滔悉。 我是一名探鬼主播,決...
    沈念sama閱讀 40,447評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼单绑,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼回官!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起搂橙,我...
    開封第一講書人閱讀 39,349評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤歉提,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后区转,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體苔巨,經(jīng)...
    沈念sama閱讀 45,820評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,990評(píng)論 3 337
  • 正文 我和宋清朗相戀三年废离,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了侄泽。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片痊硕。...
    茶點(diǎn)故事閱讀 40,127評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡卧抗,死狀恐怖盒齿,靈堂內(nèi)的尸體忽然破棺而出予颤,到底是詐尸還是另有隱情,我是刑警寧澤葛虐,帶...
    沈念sama閱讀 35,812評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響窥妇,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜娩践,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,471評(píng)論 3 331
  • 文/蒙蒙 一活翩、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧翻伺,春花似錦材泄、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,017評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春旦事,著一層夾襖步出監(jiān)牢的瞬間魁巩,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,142評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工姐浮, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留谷遂,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,388評(píng)論 3 373
  • 正文 我出身青樓卖鲤,卻偏偏與公主長(zhǎng)得像肾扰,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子蛋逾,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,066評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容