Android-插件化技术之我也来入个门-DexClassLoader加载apk,反射调用插件方法

最近完全投入Android开发一年左右了,中间也是一直补知识。到现在,还是补了蛮多的。 布局上用约束布局很爽,应该没啥大问题。 负责的布局,rv多type用的多,另外阿里的Vlayout也有尝试,还有一些其他框架,有看过一些三方框架源码,貌似也是多布局的封装,还蛮骚的样子。自定义View之前搞过,流程基本ok,问题不会太大。然后到了后面自己封装了弹窗库,新项目也用到了(近期弹窗计划正在针对地区选择进行封装,封装后正好下一个版本迭代用上),另外Android公共组件库正在考虑中,因为做了几个项目,基本很多控件都是类似的配置,而且有些还是很重复的操作,所以打算再搞一个公共组件库(当然其中包括涉及到自定义View、方便用户配置)。简单回味下....

然后一方面小萌新再看一些源码,一方面打算抽点时间再深入下其他方面,比如插件化、热修复等,想想还是蛮重要的勒!

插件化的原理相关介绍:

1. 通过DexClassLoader加载。

2. 代理模式添加生命周期。

3. Hook思想跳过清单验证。

好吧,先尝试实践下DexClassLoader加载吧,参考网友的操作我们来过一下流程! 后面就开始着手做一些较深入的分析,顺便结合相关官方资料来加深印象!

Tips: DexClassLoader.loadClass()加载后可以如下方式调用插件的方法

//通过反射调用插件的代码

//通过接口调用插件的代码(其中包括较为完善的面向切面编程调用插件的方法)

**A. **试试反射的方式:

1. 创建工程

image

2. 新建一个Module- plugin

image
image

3. 然后plugin模块下新建一个被调用的方法,比如 PluginTest.java, 并提供如下操作

   public class PluginTest {
    private String feature = "不帅";

    public String getFeature() {
        return feature;
    }

    public void setFeature(String feature) {
        this.feature = feature;
    }
}

4. 然后打包这个模块为apk

image
image

5. 将plugin下的apk拷贝到app模块下的assets目录下

image

6. 搞工具类将assets目录下的plugin-debug.apk拷贝到应用目录下,比如/data/user/0/popeeee.hl.com.plugin/files/Download/下,这样可以避免还需要动态申请存储权限的问题

image

7. 然后就可以进行拷贝操作了哟,成功后进行apk的装载

import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import popeeee.hl.com.plugin.utils.FileUtil;
import popeeee.hl.com.plugin.utils.SystemUtils;

public class MainActivity extends AppCompatActivity {
    private String pluginApkName = "plugin-debug.apk";  ///< 插件apk名称
    private String apkPath;         ///< apk存储路径
    private String apkDexPath;      ///< apk解压dex的目录、和apk存放路径为一个路径
    private DexClassLoader dexClassLoader;

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

        ///< 获取apk准备存储的应用本地缓存路径
        this.apkDexPath = SystemUtils.getCacheDirectory(this, Environment.DIRECTORY_DOWNLOADS).getPath();
        ///< 拷贝assets下的plugin-debug.apk到apkPath目录并获取实际路径
        this.apkPath = FileUtil.copyFilesFromAssets(this, pluginApkName, apkDexPath);
        ///< 加载apk并获取DexClassLoader对象
        this.dexClassLoader = new DexClassLoader(apkPath, apkDexPath, null, this.getClassLoader());
    }
}

8. 给当前控件添加一个点击事件,然后点击通过DexClassLoader.loadClass()加载插件对应的类,然后通过反射获取对应的方法进行调用, 之前关于反射的学习MonkeyLei:Android-自定义注解-控件注解

   /**
     * 默认hello world文本框添加点击事件 android:onClick="CallPlugin"
     * @param view
     */
    public void CallPlugin(View view) {
        try {
            ///< 加载插件的类(插件的包名.类名)
            Class<?> mClass = dexClassLoader.loadClass("popeeee.hl.com.plugin.PluginTest");

            ///< 获取类的实例
            Object beanObject = mClass.newInstance();

            ///< 然后通过反射获取对应的方法
            Method setFeatureMethod = mClass.getMethod("setFeature", String.class);
            setFeatureMethod.setAccessible(true);
            Method getFeatureMethod = mClass.getMethod("getFeature");
            getFeatureMethod.setAccessible(true);

            ///< 然后执行对应方法进行相关设置和获取
            setFeatureMethod.invoke(beanObject, "丑的不行呀!");
            String feature = (String) getFeatureMethod.invoke(beanObject);

            ///< 然后本地进行一些提示等操作
            Toast.makeText(this, feature, Toast.LENGTH_SHORT).show();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }

9. 当点击hello world后就可以看见回调信息了呀。。。

image

以上方式加载过程都ok。 不过很多人都是把拷贝apk放到如下地方进行调用其实拷贝很快的,不一定要放到这里?ContextWrapper类的源码,ContextWrapper中有一个attachBaseContext()方法,这个方法会将传入的一个Context参数赋值给mBase对象,之后mBase对象就有值了。

Application中在onCreate()方法里去初始化各种全局的变量数据是一种比较推荐的做法,但是如果你想把初始化的时间点提前到极致,也可以去重写attachBaseContext()方法,同时加载apk时进行一个简单判断:

    @Override
    protected void attachBaseContext(Context newBase) {
        super.attachBaseContext(newBase);

        ///< 获取apk准备存储的应用本地缓存路径
        this.apkDexPath = SystemUtils.getCacheDirectory(this, Environment.DIRECTORY_DOWNLOADS).getPath();
        ///< 拷贝assets下的plugin-debug.apk到apkPath目录并获取实际路径
        this.apkPath = FileUtil.copyFilesFromAssets(this, pluginApkName, apkDexPath);
    }

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

        ///< 判断apk是否存在
        File file = new File(apkPath);
        if (!file.exists()){
            Toast.makeText(this, "文件不存在", Toast.LENGTH_SHORT).show();
            return;
        }
        ///< 加载apk并获取DexClassLoader对象,如果有.so需要考虑第三个参数
        this.dexClassLoader = new DexClassLoader(apkPath, apkDexPath, null, this.getClassLoader());
    }

B. 上面的加载方法还是略显复杂,有点麻烦了,如果加载的对象可以直接转换为PluginTest对象岂不是妙哉!

由于app模块并没有这个PluginTeset类,所以没法这样操作,有个做法是,把插件的类复制一份到app模块,然后直接强制转换即可!试试是可以滴了....

image

这样也是没问题的。但是这样很麻烦呀,你想想,一旦插件要加个什么东西都需要拷贝一份,太烦了。 所以我们需要一个公共库,宿主和插件都依赖它,然后由它提供相关的实体类接口,这样只要都继承对应接口即可,维护起来也方便很多呢!

1. 新建一个插件库(主要是与插件对应)

image
image

2. 新建实体类对应的公共接口 PluginProvider.java

   public interface PluginProvider {
    String getFeature();

    void setFeature(String feature);
}
image

3. 宿主和插件都依赖该库,修改插件实体类继承自PluginProvider

image
image
  1. 重新打包插件apk,更新到assets目录下替换之前的插件

5. 然后宿主调用插件方式做一下改变,只需要强转为PluginProvider即可,不依赖于插件具体的实体类类型

            ///< 面向接口编程调用插件代码
            PluginProvider pluginProvider = (PluginProvider) mClass.newInstance();
            pluginProvider.setFeature("不帅么?");

            ///< 然后本地进行一些提示等操作
            Toast.makeText(this, pluginProvider.getFeature(), Toast.LENGTH_SHORT).show();
image

然后就ojbk了。

image

**C. **有时候我们希望通过回调的方式调用插件的方法,因为插件还要做很多事情才能回调给宿主(比如插件需要去下载皮肤主题资源,然后解压校验,成功后才能通知宿主进行相关设置),此时我们就采用接口编程回调的方式实现。回调我们经常用啦,问题不大哈...

1. 公共插件库中我们定义一个回调接口,并提供一个invokeCallBack(ICallBack callBack)方法. IDynamic.java

public interface IDynamic {
    void invokeCallBack(ICallBack callBack);
}

ICallBack.java

public interface ICallBack {
    void callback(PluginProvider pluginProvider);
}

PluginProvider.java

  public interface PluginProvider {
    String getFeature();

    void setFeature(String feature);
}

2. 然后插件模块就可以新建一个Dynamic 继承实现IDynamic的方法,给出回调(利用线程做一个模拟)

 import popeeee.hl.com.pluginlibrary.ICallBack;
import popeeee.hl.com.pluginlibrary.IDynamic;

public class Dynamic implements IDynamic {
    @Override
    public void invokeCallBack(final ICallBack callBack) {
                    ///< 操作获取某些信息,然后回调给宿主
                    new Thread(new Runnable() {
                        @Override
                        public void run() {
                            try {
                                Thread.sleep(3);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                PluginTest pluginTest = new PluginTest();
                pluginTest.setFeature("我来自互联网,我标志了人类的一大进步!“呸,不要脸!");
                callBack.callback(pluginTest);
            }
        }).start();
    }
}

3. 然后宿主app此时不再加载对应的实体类(因为你加载了实体类也只是自己设置,自己获取信息,没什么卵用!)。 此时我们加载Dynamic类,然后调用插件的invoke方法来请求网络等操作获取我们真实想要的数据....

记得重新打包plugin模块的apk,更新下下

然后修改下加载实体类并且进行强制转换

image
   /**
     * 默认hello world文本框添加点击事件 android:onClick="CallPlugin"
     * @param view
     */
    public void CallPlugin(View view) {
        try {
            ///< 加载插件的类(插件的包名.类名)
            Class<?> mClass = dexClassLoader.loadClass("popeeee.hl.com.plugin.Dynamic");

            /// 1\. 反射方式调用
//            ///< 获取类的实例
//            Object beanObject = mClass.newInstance();
//
//            ///< 然后通过反射获取对应的方法
//            Method setFeatureMethod = mClass.getMethod("setFeature", String.class);
//            setFeatureMethod.setAccessible(true);
//            Method getFeatureMethod = mClass.getMethod("getFeature");
//            getFeatureMethod.setAccessible(true);
//
//            ///< 然后执行对应方法进行相关设置和获取
//            setFeatureMethod.invoke(beanObject, "丑的不行呀!");
//            String feature = (String) getFeatureMethod.invoke(beanObject);

//            ///< 然后本地进行一些提示等操作
//            Toast.makeText(this, feature, Toast.LENGTH_SHORT).show();

//            /// 2\. 强制转换对应包含操作方法的对象
//            PluginTest pluginTest = (PluginTest) mClass.newInstance();
//            pluginTest.setFeature("丑的还可以呀2!");
//
//            ///< 然后本地进行一些提示等操作
//            Toast.makeText(this, pluginTest.getFeature(), Toast.LENGTH_SHORT).show();

//            ///< 面向接口编程调用插件代码
//            PluginProvider pluginProvider = (PluginProvider) mClass.newInstance();
//            pluginProvider.setFeature("不帅么?");
//
//            ///< 然后本地进行一些提示等操作
//            Toast.makeText(this, pluginProvider.getFeature(), Toast.LENGTH_SHORT).show();

            ///< 面向切面编程调用插件代码
            IDynamic iDynamic = (IDynamic) mClass.newInstance();
            iDynamic.invokeCallBack(new ICallBack() {
                @Override
                public void callback(PluginProvider pluginProvider) {
                    Looper.prepare();
                    ///< 然后本地进行一些提示等操作
                    Toast.makeText(MainActivity.this, pluginProvider.getFeature(), Toast.LENGTH_SHORT).show();
                    Looper.loop();
                }
            });
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        }
    }

这样就可以了

image

到这里插件的入门算是有所了解,另外自己亲自实践了一把,感觉还是不一样的。另外还有插件的两个入门点,一个是插件资源的加载,一个是插件的Activity的加载启动。这个两个小萌新要后面再搞。

搞的前提:1. 小萌新要去了解资源加载相关的机制,原理,源码的解读 2. 同样Activity的加载也是需要解读一些源码方可深入些。 另外如果对ClassLoader还在陌生的话,有必要去看下官方api,做一个解读了....

Demo下载地址还是贴下吧,万一需要了 https://gitee.com/heyclock/doc/blob/master/PluginTest/PluginTest.zip

先到这,贴几个我觉得不错的文章,共勉之,一起加油, 很多东西还是要自己实践...还得有自己理解!

Android插件化技术入门

Android插件化入门指南

Android插件化——资源加载

https://blog.csdn.net/liangfeng093/article/details/78120803

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

推荐阅读更多精彩内容