超級(jí)簡(jiǎn)單的熱修復(fù)實(shí)現(xiàn)步驟詳解

??????看了熱修復(fù)好久,大多文章都講的原理,沒(méi)有發(fā)現(xiàn)有詳細(xì)的實(shí)現(xiàn)步驟,所以自己看完視頻之后對(duì)自己的實(shí)現(xiàn)步驟進(jìn)行了記錄總結(jié),方便自己記憶,也希望對(duì)讀到這篇文章的童鞋有所幫助,因?yàn)閷?duì)熱修復(fù)愿意很多大牛的文章已有詳細(xì)的記錄,此處不再班門(mén)弄斧,主要記錄實(shí)現(xiàn)步驟,文章前部分會(huì)說(shuō)明具體實(shí)現(xiàn),文章后部分有完整代碼,后期會(huì)附上git倉(cāng)庫(kù)地址(ps:此處不注重對(duì)權(quán)限的管理,所以請(qǐng)?jiān)谶\(yùn)行項(xiàng)目時(shí)確保對(duì)此APP的讀寫(xiě)權(quán)限已經(jīng)開(kāi)啟)
??????文章大概分為
??????1.編寫(xiě)相應(yīng)工具方法,并編寫(xiě)使程序崩潰的錯(cuò)誤方法進(jìn)行運(yùn)行測(cè)試
??????2.將崩潰方法修復(fù),并手動(dòng)打出修復(fù)的dex包,復(fù)制到手機(jī)的sd卡目錄
??????3.將程序修改為崩潰方法,進(jìn)行重新運(yùn)行,并進(jìn)行修復(fù)

一.在模塊目錄的build.gradle文件添加相應(yīng)配置

??????1.在dependencies中添加如下代碼

dependencies {
    compile 'com.android.support:multidex:1.0.1'
}

??????2.在android的defaultConfig中添加如下代碼

defaultConfig {
        multiDexEnabled true
    }

??????3.在buildTypes的release下添加如下代碼

 release {
            minifyEnabled true
        }

二.在自定義的Application中添加相應(yīng)方法

??????重寫(xiě)attachBaseContext并添加如下代碼

@Override
    protected void attachBaseContext(Context base) {
        MultiDex.install(base);
        super.attachBaseContext(base);
    }

三.創(chuàng)建MyConstants類(lèi),用于保存應(yīng)用中用到的常量

??????1.聲明保存文件的文件名

    /**
     * 創(chuàng)建文件時(shí)的文件名
     */
    public static String DEX_DIR = "odex";

四.創(chuàng)建FixDexUtils工具類(lèi),用于編寫(xiě)熱修復(fù)用到的方法

??????1.聲明loadedDex用于存儲(chǔ)讀取到的dex文件

private static HashSet<File> loadedDex = new HashSet<>();
    static {
        loadedDex.clear();
    }

??????2.添加通過(guò)反射給指定類(lèi)的指定屬性賦值的方法

private static void setFiled(Object obj,Class<?> cl,String field,Object value) throws Exception{
        Field localField = cl.getDeclaredField(field);
        localField.setAccessible(true);
        localField.set(obj,value);
    }

??????3.添加通過(guò)反射獲取指定類(lèi)的指定值的方法

private static Object getFiled(Object obj,Class<?> cl,String field) throws Exception{
        Field localField = cl.getDeclaredField(field);
        localField.setAccessible(true);
        return localField.get(obj);
    }

??????4.添加得到類(lèi)加載器的pathList的方法

 private static Object getPathList(Object baseDexClassLoader) throws Exception{
        return getFiled(baseDexClassLoader,Class.forName("dalvik.system.BaseDexClassLoader"),"pathList");
    }

??????5.添加得到dexElements的方法

private static Object getDexElements(Object obj) throws Exception{
        return getFiled(obj,obj.getClass(),"dexElements");
    }

??????6.添加合并兩個(gè)數(shù)組的方法

private static Object conbineArray(Object arrayLbs,Object arrayRhs){
        Class<?> localClass = arrayLbs.getClass().getComponentType();
        int i = Array.getLength(arrayLbs);
        int j = i + Array.getLength(arrayRhs);
        Object result = Array.newInstance(localClass,j);
        for(int k = 0;k < j; ++k){
            if(k < i){
                Array.set(result,k,Array.get(arrayLbs,k));
            }else{
                Array.set(result,k,Array.get(arrayRhs,k - i));
            }
        }
        return result;
    }

??????7.將項(xiàng)目的dex和已修復(fù)的dex合并,并賦值給類(lèi)加載器,進(jìn)行熱修復(fù)

private static void doDexInject(final Context appContext,File filesDir,HashSet<File> loadedDex){
        try {
            String optiondexDir = filesDir.getAbsolutePath()+File.separator+"opt_dex";
            File fopt = new File(optiondexDir);
            if(!fopt.exists()){
                fopt.mkdir();
            }
            //1.加載應(yīng)用程序的dex
            PathClassLoader pathLoader = (PathClassLoader)appContext.getClassLoader();
            for(File dex : loadedDex){
                //2.加載指定的修復(fù)的dex文件
                DexClassLoader classLoader = new DexClassLoader(
                        dex.getAbsolutePath(),
                        fopt.getAbsolutePath(),
                        null,
                        pathLoader);
                //3.合并
                Object dexObj = getPathList(classLoader);
                Object pathObj = getPathList(pathLoader);
                Object mDexElementsList = getDexElements(dexObj);
                Object pathDexElementsList = getDexElements(pathObj);
                //將兩個(gè)list合并為一個(gè)
                Object dexElements = conbineArray(mDexElementsList,pathDexElementsList);
                //重寫(xiě)給PathList里面的Element[] dexElements賦值
                Object pathList = getPathList(pathLoader);
                setFiled(pathList,pathList.getClass(),"dexElements",dexElements);
            }
        }catch (Exception e){}
    }

??????8.添加加載sd卡中的修復(fù)文件,并進(jìn)行合并,完成修復(fù)的方法

public static void loadFixedDex(Context context){
        if(null == context){
            return;
        }
        //遍歷所有的修復(fù)的dex
        File fileDir = context.getDir(MyConstants.DEX_DIR,Context.MODE_PRIVATE);
        File[] listFiles = fileDir.listFiles();
        for(File file : listFiles){
            if(file.getName().startsWith("classes") || file.getName().endsWith(".dex")){
                loadedDex.add(file);//先將補(bǔ)丁文件放到一個(gè)集合里,然后再進(jìn)行合并
            }
        }
        //dex合并之前的dex
        doDexInject(context,fileDir,loadedDex);
    }

五.創(chuàng)建MyTestClass文件,用于編寫(xiě)測(cè)試方法,即會(huì)造成程序崩潰的方法,并后期對(duì)其進(jìn)行修復(fù)

public void testFix(Context context){
        int i = 10;
        int a = 0;
        Toast.makeText(context,"shit:" + i/a,Toast.LENGTH_SHORT).show();
    };

六.在MainActivity中編寫(xiě)相關(guān)方法

??????1.添加兩個(gè)btn,分別為test,和fix,test用于執(zhí)行測(cè)試方法test(View view),fix用于執(zhí)行修復(fù)方法fix(View view)
??????2.添加測(cè)試方法(即使程序崩潰的方法)

public void test(View view) {
        MyTestClass myTestClass = new MyTestClass();
        myTestClass.testFix(this);
    }

??????3.添加修復(fù)方法

 /**
     * 修復(fù)的方法
     * @param view
     */
    public void fix(View view) {
        fixBug();
    }

    private void fixBug() {
        //目錄 data/data/packageName/odex
        File fileDir = getDir(MyConstants.DEX_DIR, Context.MODE_PRIVATE);
        //該目錄下放置修復(fù)好的dex文件
        String name = "classes2.dex";
        String filePath = fileDir.getAbsolutePath() + File.separator + name;
        File file = new File(filePath);
        if(file.exists()){
            file.delete();
        }
        //搬家:把下載好的在sd卡里面的修復(fù)了的classes2.dex復(fù)制到應(yīng)用目錄
        InputStream is = null;
        FileOutputStream os = null;
        try {
            //復(fù)制并粘貼文件
            is = new FileInputStream(Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+name);
            os = new FileOutputStream(filePath);
            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1){
                os.write(buffer,0,len);
            }

            //粘貼完文件
            File f = new File(filePath);
            if(f.exists()){//文件從sk卡賦值到應(yīng)用運(yùn)行目錄下,成功則toast提示
                Toast.makeText(this,"dex重寫(xiě)成功",Toast.LENGTH_SHORT).show();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

七.在AndroidManifest.xml中添加讀寫(xiě)權(quán)限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>

??????tips:因?yàn)樵谛迯?fù)過(guò)程中涉及文件的拷貝,需聲明該權(quán)限,因權(quán)限不是此文章的關(guān)注點(diǎn),故忽略申請(qǐng)權(quán)限的部分,請(qǐng)手動(dòng)開(kāi)啟該測(cè)試app的讀寫(xiě)權(quán)限,并在權(quán)限管理中進(jìn)一步確認(rèn)

八.在自定義的Application中添加調(diào)用修復(fù)文件的方法

    @Override
    protected void attachBaseContext(Context base) {
        MultiDex.install(base);
        //加載修復(fù)文件,并進(jìn)行相應(yīng)操作
        FixDexUtils.loadFixedDex(base);
        super.attachBaseContext(base);
    }

九.手動(dòng)打包修復(fù)bug的dex文件

??????此時(shí)運(yùn)行程序點(diǎn)擊測(cè)試按鈕,執(zhí)行錯(cuò)誤代碼,會(huì)造成程序崩潰,要修復(fù)問(wèn)題,需先將測(cè)試代碼修改為沒(méi)有問(wèn)題的代碼,并手動(dòng)打出dex包,進(jìn)行修復(fù)
??????1.調(diào)整MyTestClass中的testFix方法為正確代碼

 public void testFix(Context context){
        int i = 10;
        int a = 1;
        Toast.makeText(context,"shit:" + i/a,Toast.LENGTH_SHORT).show();
    };

??????2.卸載手機(jī)安裝的應(yīng)用,然后重新安裝應(yīng)用(即修復(fù)testFix方法的應(yīng)用)

??????3.找到編譯后的MyTestClass.class文件(在項(xiàng)目名\app\build\intermediates\classes\debug\com\hotfixdemo下,com\hotfixdemo為項(xiàng)目的包名目錄,我的文件路徑為D:\Project\HotFixDemo\app\build\intermediates\classes\debug\com\hotfixdemo)
MyTestClass.class文件路徑.PNG

??????4.將MyTestClass.class文件連同包目錄拷貝到一個(gè)文件夾,如我的為桌面dex文件夾
拷貝文件完成圖.png

??????5.配置dx.bat打包工具,并手動(dòng)打包生成classes2.dex(因?yàn)轫?xiàng)目中指定了該文件名稱(chēng),所以此處應(yīng)保持名稱(chēng)一致)
????????????(1)在project模式下,右鍵項(xiàng)目名稱(chēng)-Open Moudle Setting查看項(xiàng)目編譯時(shí)使用的build tools版本,此處為26.0.2


查看tools版本.png

????????????(2)在sdk所在目錄找到相應(yīng)build tools版本下de.bat所在的路徑,復(fù)制該地址,并配置到環(huán)境變量,此處為D:\ProgramFiles\sdk\andsdk\build-tools\26.0.2
dx.bat所在路徑.png

配置dx.bat的環(huán)境變量.png

????????????(3)打開(kāi)cmd,切換到用于保存MyTestClass.class的(即九-4步驟中的路徑,此處為桌面dex文件夾)路徑,執(zhí)行命令行:dx --dex --output=存放生成的dex文件所在的路徑\classes2.dex MyTestClass.class包所在路徑,此處完整命令行為
dx --dex --output=C:\Users\cuixiaoxiao\Desktop\classes2.dex C:\Users\cuixiaoxiao\Desktop\dex

執(zhí)行完可在相應(yīng)輸入目錄(此處為桌面)找到classes2.dex文件


cmd手動(dòng)編譯產(chǎn)生dex文件.png

????????????(4)將編譯產(chǎn)生的classes2.dex文件復(fù)制到手機(jī)的sd卡目錄下


復(fù)制文件到sd卡.png

十.將項(xiàng)目改為使程序崩潰的測(cè)試代碼,進(jìn)行熱修復(fù)的測(cè)試

??????1.將MyTestClass里面的testFix方法修改為使程序崩潰的方法

  public void testFix(Context context){
        int i = 10;
        int a = 0;
        Toast.makeText(context,"shit:" + i/a,Toast.LENGTH_SHORT).show();
    };

??????2.卸載手機(jī)的應(yīng)用,重新安裝應(yīng)用(即會(huì)使程序崩潰的應(yīng)用)
??????3.請(qǐng)先手動(dòng)確認(rèn)下,開(kāi)啟了該應(yīng)用對(duì)sd的讀寫(xiě)權(quán)限


開(kāi)啟sd卡讀寫(xiě)權(quán)限.jpg

??????4.此時(shí)點(diǎn)擊test按鈕,程序崩潰


未修復(fù)前.png

??????5.點(diǎn)擊fix按鈕,進(jìn)行熱修復(fù)
進(jìn)行文件修復(fù).jpg

??????6.退出應(yīng)用,并清除后臺(tái)運(yùn)行,重新開(kāi)啟應(yīng)用

??????7.此時(shí)點(diǎn)擊test按鈕,正常運(yùn)行,程序不再崩潰,則熱修復(fù)成功


修復(fù)成功,點(diǎn)擊test不再崩潰.jpg

附錄

1.model的build.gradle文件

apply plugin: 'com.android.application'

android {
    compileSdkVersion 26
    defaultConfig {
        applicationId "com.hotfixdemo"
        minSdkVersion 16
        targetSdkVersion 26
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
        multiDexEnabled true
    }
    buildTypes {
        release {
            minifyEnabled true
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
    buildToolsVersion '26.0.2'
}

dependencies {
    implementation fileTree(include: ['*.jar'], dir: 'libs')
    implementation 'com.android.support:appcompat-v7:26.1.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.0'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
    compile 'com.android.support:multidex:1.0.1'
}

2.自定義的Application,此處為MyApplication

public class MyApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
    }

    @Override
    protected void attachBaseContext(Context base) {
        MultiDex.install(base);
        //加載修復(fù)文件,并進(jìn)行相應(yīng)操作
        FixDexUtils.loadFixedDex(base);
        super.attachBaseContext(base);
    }
}

3.常量存儲(chǔ)類(lèi)MyConstants

public class MyConstants {
    /**
     * 創(chuàng)建文件時(shí)的文件名
     */
    public static String DEX_DIR = "odex";
}

4.熱修復(fù)工具類(lèi)

import android.content.Context;

import java.io.File;
import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.util.HashSet;

import dalvik.system.DexClassLoader;
import dalvik.system.PathClassLoader;

/**
 * Created by cuixiaoxiao on 2018/5/8.
 */

public class FixDexUtils {
    private static HashSet<File> loadedDex = new HashSet<>();
    static {
        loadedDex.clear();
    }

    /**
     * 加載修復(fù)文件
     * @param context
     */
    public static void loadFixedDex(Context context){
        if(null == context){
            return;
        }
        //遍歷所有的修復(fù)的dex
        File fileDir = context.getDir(MyConstants.DEX_DIR,Context.MODE_PRIVATE);
        File[] listFiles = fileDir.listFiles();
        for(File file : listFiles){
            if(file.getName().startsWith("classes") || file.getName().endsWith(".dex")){
                loadedDex.add(file);//先將補(bǔ)丁文件放到一個(gè)集合里,然后再進(jìn)行合并
            }
        }
        //dex合并之前的dex
        doDexInject(context,fileDir,loadedDex);
    }

    /**
     * 將項(xiàng)目的dex和已經(jīng)修復(fù)的dex合并,并賦值給類(lèi)加載器,進(jìn)行熱修復(fù)
     * @param appContext
     * @param filesDir
     * @param loadedDex
     */
    private static void doDexInject(final Context appContext,File filesDir,HashSet<File> loadedDex){
        try {
            String optiondexDir = filesDir.getAbsolutePath()+File.separator+"opt_dex";
            File fopt = new File(optiondexDir);
            if(!fopt.exists()){
                fopt.mkdir();
            }
            //1.加載應(yīng)用程序的dex
            PathClassLoader pathLoader = (PathClassLoader)appContext.getClassLoader();
            for(File dex : loadedDex){
                //2.加載指定的修復(fù)的dex文件
                DexClassLoader classLoader = new DexClassLoader(
                        dex.getAbsolutePath(),
                        fopt.getAbsolutePath(),
                        null,
                        pathLoader);
                //3.合并
                Object dexObj = getPathList(classLoader);
                Object pathObj = getPathList(pathLoader);
                Object mDexElementsList = getDexElements(dexObj);
                Object pathDexElementsList = getDexElements(pathObj);
                //將兩個(gè)list合并為一個(gè)
                Object dexElements = conbineArray(mDexElementsList,pathDexElementsList);
                //重寫(xiě)給PathList里面的Element[] dexElements賦值
                Object pathList = getPathList(pathLoader);
                setFiled(pathList,pathList.getClass(),"dexElements",dexElements);
            }
        }catch (Exception e){}
    }

    /**
     * 得到類(lèi)加載器的pathList
     * @param baseDexClassLoader
     * @return
     * @throws Exception
     */
    private static Object getPathList(Object baseDexClassLoader) throws Exception{
        return getFiled(baseDexClassLoader,Class.forName("dalvik.system.BaseDexClassLoader"),"pathList");
    }

    /**
     * 通過(guò)反射給指定類(lèi)的指定屬性賦值
     * @param obj
     * @param cl
     * @param field
     * @param value
     * @throws Exception
     */
    private static void setFiled(Object obj,Class<?> cl,String field,Object value) throws Exception{
        Field localField = cl.getDeclaredField(field);
        localField.setAccessible(true);
        localField.set(obj,value);
    }

    /**
     * 通過(guò)反射調(diào)用指定類(lèi)的方法
     * @param obj
     * @param cl
     * @param field
     * @return
     * @throws Exception
     */
    private static Object getFiled(Object obj,Class<?> cl,String field) throws Exception{
        Field localField = cl.getDeclaredField(field);
        localField.setAccessible(true);
        return localField.get(obj);
    }

    /**
     * 得到dexElements
     * @param obj
     * @return
     * @throws Exception
     */
    private static Object getDexElements(Object obj) throws Exception{
        return getFiled(obj,obj.getClass(),"dexElements");
    }

    /**
     * 合并兩個(gè)數(shù)組
     * @param arrayLbs
     * @param arrayRhs
     * @return
     */
    private static Object conbineArray(Object arrayLbs,Object arrayRhs){
        Class<?> localClass = arrayLbs.getClass().getComponentType();
        int i = Array.getLength(arrayLbs);
        int j = i + Array.getLength(arrayRhs);
        Object result = Array.newInstance(localClass,j);
        for(int k = 0;k < j; ++k){
            if(k < i){
                Array.set(result,k,Array.get(arrayLbs,k));
            }else{
                Array.set(result,k,Array.get(arrayRhs,k - i));
            }
        }
        return result;
    }
}

5.測(cè)試類(lèi)MyTestClass

public class MyTestClass {
    /**
     * 測(cè)試方法,會(huì)導(dǎo)致程序崩潰
     * @param context
     */
    public void testFix(Context context){
        int i = 10;
        int a = 0;
        Toast.makeText(context,"shit:" + i/a,Toast.LENGTH_SHORT).show();
    };
}

6.存在測(cè)試按鈕的activity

import android.content.Context;
import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.InputStream;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    /**
     * 測(cè)試方法(即會(huì)導(dǎo)致應(yīng)用崩潰)
     * @param view
     */
    public void test(View view) {
        MyTestClass myTestClass = new MyTestClass();
        myTestClass.testFix(this);
    }

    /**
     * 修復(fù)的方法
     * @param view
     */
    public void fix(View view) {
        fixBug();
    }

    private void fixBug() {
        //目錄 data/data/packageName/odex
        File fileDir = getDir(MyConstants.DEX_DIR, Context.MODE_PRIVATE);
        //該目錄下放置修復(fù)好的dex文件
        String name = "classes2.dex";
        String filePath = fileDir.getAbsolutePath() + File.separator + name;
        File file = new File(filePath);
        if(file.exists()){
            file.delete();
        }
        //搬家:把下載好的在sd卡里面的修復(fù)了的classes2.dex復(fù)制到應(yīng)用目錄
        InputStream is = null;
        FileOutputStream os = null;
        try {
            //復(fù)制并粘貼文件
            is = new FileInputStream(Environment.getExternalStorageDirectory().getAbsolutePath()+File.separator+name);
            os = new FileOutputStream(filePath);
            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1){
                os.write(buffer,0,len);
            }

            //粘貼完文件
            File f = new File(filePath);
            if(f.exists()){//文件從sk卡賦值到應(yīng)用運(yùn)行目錄下,成功則toast提示
                Toast.makeText(this,"dex重寫(xiě)成功",Toast.LENGTH_SHORT).show();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

7.AndroidManifest

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.hotfixdemo">
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
    <application
        android:name=".MyApplication"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

文章為自己總結(jié)所得,如有錯(cuò)誤的地方歡迎指出,會(huì)即使修改,有問(wèn)題的地方,歡迎提問(wèn),如您覺(jué)得有可借鑒的地方,歡迎轉(zhuǎn)載,請(qǐng)注明出處,多謝多謝

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容