美团热修复的使用

美团热修复的使用

如何集成?

  1. 在App的build.gradle,加入如下依赖

    apply plugin: 'com.android.application'
    //制作补丁时将这个打开,auto-patch-plugin紧跟着com.android.application
    //apply plugin: 'auto-patch-plugin'
    apply plugin: 'robust'
    
    在dependencies里加入
    compile 'com.meituan.robust:robust:0.4.75'
    
    
  2. 在工程的build.gradle中,加入classpath

    buildscript {
        repositories {
            jcenter()
        }
        dependencies {
             classpath 'com.meituan.robust:gradle-plugin:0.4.75'
             classpath 'com.meituan.robust:auto-patch-plugin:0.4.75'
       }
    } 
    
    
  3. 在app目录下,配置robust.xml文件,具体配置见官方demo https://github.com/Meituan-Dianping/Robust/blob/master/app/robust.xml,以下是我的配置

    <?xml version="1.0" encoding="utf-8"?>
    <resources>
    
        <switch>
            <!--true代表打开Robust,请注意即使这个值为true,Robust也默认只在Release模式下开启-->
            <!--false代表关闭Robust,无论是Debug还是Release模式都不会运行robust-->
            <turnOnRobust>true</turnOnRobust>
            <!--<turnOnRobust>false</turnOnRobust>-->
    
            <!--是否开启手动模式,手动模式会去寻找配置项patchPackname包名下的所有类,自动的处理混淆,然后把patchPackname包名下的所有类制作成补丁-->
            <!--这个开关只是把配置项patchPackname包名下的所有类制作成补丁,适用于特殊情况,一般不会遇到-->
            <!--<manual>true</manual>-->
            <manual>false</manual>
    
            <!--是否强制插入插入代码,Robust默认在debug模式下是关闭的,开启这个选项为true会在debug下插入代码-->
            <!--但是当配置项turnOnRobust是false时,这个配置项不会生效-->
            <!--<forceInsert>true</forceInsert>-->
            <forceInsert>false</forceInsert>
    
            <!--是否捕获补丁中所有异常,建议上线的时候这个开关的值为true,测试的时候为false-->
            <catchReflectException>true</catchReflectException>
            <!--<catchReflectException>false</catchReflectException>-->
    
            <!--是否在补丁加上log,建议上线的时候这个开关的值为false,测试的时候为true-->
            <!--<patchLog>true</patchLog>-->
            <patchLog>false</patchLog>
    
            <!--项目是否支持progaurd-->
            <proguard>false</proguard>
            <!--<proguard>false</proguard>-->
    
            <!--项目是否支持ASM进行插桩,默认使用ASM,推荐使用ASM,Javaassist在容易和其他字节码工具相互干扰-->
            <useAsm>true</useAsm>
            <!--<useAsm>false</useAsm>-->
        </switch>
    
        <!--需要热补的包名或者类名,这些包名下的所有类都被会插入代码-->
        <!--这个配置项是各个APP需要自行配置,就是你们App里面你们自己代码的包名,
        这些包名下的类会被Robust插入代码,没有被Robust插入代码的类Robust是无法修复的-->
        <packname name="hotfixPackage">
            <name>com.client.api</name>
        </packname>
    
        <!--不需要Robust插入代码的包名,Robust库不需要插入代码,如下的配置项请保留,还可以根据各个APP的情况执行添加-->
        <exceptPackname name="exceptPackage">
            <name>com.meituan.robust</name>
            <name>com.meituan.sample.extension</name>
        </exceptPackname>
    
        <!--补丁的包名,请保持和类PatchManipulateImp中fetchPatchList方法中设置的补丁类名保持一致( setPatchesInfoImplClassFullName("com.meituan.robust.patch.PatchesInfoImpl")),
        各个App可以独立定制,需要确保的是setPatchesInfoImplClassFullName设置的包名是如下的配置项,类名必须是:PatchesInfoImpl-->
        <patchPackname name="patchPackname">
            <name>com.kira.drivetest</name>
        </patchPackname>
    
        <!--自动化补丁中,不需要反射处理的类,这个配置项慎重选择-->
        <noNeedReflectClass name="classes no need to reflect">
    
        </noNeedReflectClass>
    </resources>
    

    这里面最重要的是packname和patchPackname这两个的配置。其他的基本可以不动。

    packname里面的内容写上你要修复代码处的包名。patchPackname里的内容就写上继承了PatchManipulate类所在的位置,以下是我的配置

    public class PatchManipulateImp extends PatchManipulate {
        /***
         * connect to the network ,get the latest patches
         * l联网获取最新的补丁
         * @param context
         *
         * @return
         */
        @Override
        protected List<Patch> fetchPatchList(Context context) {
            //将app自己的robustApkHash上报给服务端,服务端根据robustApkHash来区分每一次apk build来给app下发补丁
            //apkhash is the unique identifier for  apk,so you cannnot patch wrong apk.
            String robustApkHash = RobustApkHashUtils.readRobustApkHash(context);
            Log.w("robust","robustApkHash :" + robustApkHash);
            //connect to network to get patch list on servers
            //在这里去联网获取补丁列表
            Patch patch = new Patch();
            patch.setName("123");
            //we recommend LocalPath store the origin patch.jar which may be encrypted,while TempPath is the true runnable jar
            //LocalPath是存储原始的补丁文件,这个文件应该是加密过的,TempPath是加密之后的,TempPath下的补丁加载完毕就删除,保证安全性
            //这里面需要设置一些补丁的信息,主要是联网的获取的补丁信息。重要的如MD5,进行原始补丁文件的简单校验,以及补丁存储的位置,这边推荐把补丁的储存位置放置到应用的私有目录下,保证安全性
            patch.setLocalPath(Environment.getExternalStorageDirectory().getPath()+ File.separator+"robust"+File.separator + "patch");
    
            //setPatchesInfoImplClassFullName 设置项各个App可以独立定制,需要确保的是setPatchesInfoImplClassFullName设置的包名是和xml配置项patchPackname保持一致,而且类名必须是:PatchesInfoImpl
            //请注意这里的设置
            patch.setPatchesInfoImplClassFullName("com.kira.drivetest.PatchesInfoImpl");
            List  patches = new ArrayList<Patch>();
            patches.add(patch);
            return patches;
        }
    
        /**
         *
         * @param context
         * @param patch
         * @return
         *
         * you can verify your patches here
         */
        @Override
    
        protected boolean verifyPatch(Context context, Patch patch) {
            //do your verification, put the real patch to patch
            //放到app的私有目录
            patch.setTempPath(context.getCacheDir()+ File.separator+"robust"+File.separator + "patch");
            //in the sample we just copy the file
            try {
                copy(patch.getLocalPath(), patch.getTempPath());
            }catch (Exception e){
                e.printStackTrace();
                throw new RuntimeException("copy source patch to local patch error, no patch execute in path "+patch.getTempPath());
            }
    
            return true;
        }
        public void copy(String srcPath,String dstPath) throws IOException {
            File src=new File(srcPath);
            if(!src.exists()){
                throw new RuntimeException("source patch does not exist ");
            }
            File dst=new File(dstPath);
            if(!dst.getParentFile().exists()){
                dst.getParentFile().mkdirs();
            }
            InputStream in = new FileInputStream(src);
            try {
                OutputStream out = new FileOutputStream(dst);
                try {
                    // Transfer bytes from in to out
                    byte[] buf = new byte[1024];
                    int len;
                    while ((len = in.read(buf)) > 0) {
                        out.write(buf, 0, len);
                    }
                } finally {
                    out.close();
                }
            } finally {
                in.close();
            }
        }
        /**
         *
         * @param patch
         * @return
         *
         * you may download your patches here, you can check whether patch is in the phone
         */
        @Override
        protected boolean ensurePatchExist(Patch patch) {
            return true;
        }
    }
    

    请注意,patch.setPatchesInfoImplClassFullName();这个调用,里面的参数写上当前这个类所在的包名及类名。

  4. 实现RobustCallBack接口,以下是具体实现

public class RobustCallBackSample implements RobustCallBack {

    @Override
    public void onPatchListFetched(boolean result, boolean isNet, List<Patch> patches) {
        Log.d("RobustCallBack", "onPatchListFetched result: " + result);
        Log.d("RobustCallBack", "onPatchListFetched isNet: " + isNet);
        for (Patch patch : patches) {
            Log.d("RobustCallBack", "onPatchListFetched patch: " + patch.getName());
        }
    }

    @Override
    public void onPatchFetched(boolean result, boolean isNet, Patch patch) {
        Log.d("RobustCallBack", "onPatchFetched result: " + result);
        Log.d("RobustCallBack", "onPatchFetched isNet: " + isNet);
        Log.d("RobustCallBack", "onPatchFetched patch: " + patch.getName());
    }

    @Override
    public void onPatchApplied(boolean result, Patch patch) {
        Log.d("RobustCallBack", "onPatchApplied result: " + result);
        Log.d("RobustCallBack", "onPatchApplied patch: " + patch.getName());

    }

    @Override
    public void logNotify(String log, String where) {
        Log.d("RobustCallBack", "logNotify log: " + log);
        Log.d("RobustCallBack", "logNotify where: " + where);
    }

    @Override
    public void exceptionNotify(Throwable throwable, String where) {
        Log.e("RobustCallBack", "exceptionNotify where: " + where, throwable);
    }
}

这个类是关于补丁应用状况的一些回调,通知。

  1. 在需要加载补丁的地方,加上以下代码:
new PatchExecutor(this,new PatchManipulateImp(),new RobustCallBackSample()).start();
参数一:传入Context上下文
参数二:传入PatchManipulate的继承类
参数二:传入RobustCallBack的实现类

如何使用?

  1. 平时打包的时候就把app下的build.gradle里的apply plugin:'robust' 打开,屏蔽掉apply plugin: 'auto-patch-plugin'
apply plugin: 'com.android.application'
//制作补丁时将这个打开,auto-patch-plugin紧跟着com.android.application
//apply plugin: 'auto-patch-plugin'
apply plugin: 'robust'

然后将/app/build/outputs/robust里面的methodsMap.robust保存到app目录下的robust文件夹,若没有这个文件夹请创建。

  1. 修复bug的时候,将apply plugin:'auto-patch-plugin'打开,apply plugin:'robust'屏蔽。
  2. 修改代码,在改动的方法上面添加@Modify注解,对于Lambda表达式请在修改的方法里面调用RobustModify.modify()方法
 @Modify
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
     }
     //或者是被修改的方法里面调用RobustModify.modify()方法
     protected void onCreate(Bundle savedInstanceState) {
        RobustModify.modify()
        super.onCreate(savedInstanceState);
     }
  1. 在robust.xml文件中的packname字段里插入该代码所在的位置,具体到包名即可
  2. 然后执行打包命令即可,当提示

Error:Execution failed for task ':app:transformClassesWithAutoPatchTransformForRelease'.

auto patch end successfully

就代表补丁包打出来了。位置处于app/build/robust/patch.jar。补丁包可以放到公司的服务器上,然后触发加载补丁地方的代码来联网获取即可。至此,整个热修复过程完成。

常见问题

在编译补丁的时候弹出java.lang.reflect.UndeclaredThrowableException错误。

弹出这个错误的原因是,有地方的代码修改了并且标注了Modify,但是没在robust.xml文件中的packname中加上这个修改代码的所在的包名。具体的位置信息,请在编译补丁的时候查看Gradle Console中的结果,Gradle Console在AndroidStudio工具的右下角 ,EventLog的右边。

补丁无法加载成功,并且抛出copy source patch to local patch error, no patch execute in path

请注意PatchManipulate继承类里的

patch.setLocalPath(Environment.getExternalStorageDirectory().getPath()+ File.separator+"robust"+ File.separator + "patch");

这段代码是设置patch的本地路径。

在verifyPatch方法里

protected boolean verifyPatch(Context context, Patch patch) {
        //do your verification, put the real patch to patch
        //放到app的私有目录
        patch.setTempPath(context.getCacheDir()+ File.separator+"robust"+ File.separator + "patch");
        //in the sample we just copy the file
        try {
            copy(patch.getLocalPath(), patch.getTempPath());
        }catch (Exception e){
            e.printStackTrace();
            throw new RuntimeException("copy source patch to local patch error, no patch execute in path "+patch.getTempPath());
        }

        return true;
    }

这个方法里会调用copy方法,patch.getLocalPath()会帮你之前设置的路径加上.jar

public String getLocalPath() {
    return this.localPath + ".jar";
}

所以您在服务器上的补丁名字要叫patch.jar,然后patch.setLocalPath里这样设置,patch不能加jar

patch.setLocalPath(Environment.getExternalStorageDirectory().getPath()+ File.separator+"robust"+ File.separator + "patch");

,否则在copy方法里会判断这个路径不存在,因为patch.getLocalPath()里会帮你加上.jar,如果您在之前的路径上加上了jar,就变成了patch.jar.jar这样就和你下载的文件名就不一样了,所以就会抛异常了。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,874评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,151评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,270评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,137评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,116评论 5 370
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,935评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,261评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,895评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,342评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,854评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,978评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,609评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,181评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,182评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,402评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,376评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,677评论 2 344

推荐阅读更多精彩内容