通过Hook动态注入方法

问题背景

很长时间没写东西了,这段时间在做移动平台,包括组件化和插件化。这中间的坑数不胜数。
这次需要hook的地方应用背景是:
- 我有一个Base module,包含基础框架,Activity管理、网络请求、BaseActivity等的封装等等。
- 我有一个Host module,里面只有360 Replugin的宿主初始化,继承Host的app就是一个宿主。
- Base module里面的ActivityStack里有设置路由的方法,其中包括路由到插件页面。宿主跳转到插件的方法应该是这样的:

RePlugin.startActivityForResult(activity, intent, requestCode);

But,Base module里没有RePlugin这个类,只有Host里面才有,但是我非要在Base 里面调用这个方法啊,于是就考虑到了hook动态注入。

1.选择合适的hook节点

首先,上面说的跳转方法是这样的:
在Base里面,

/**
 * Created by zhouzhidong on 2017/10/24.
 * 开启插件Activity的管理类,方法无内容,需要hook
 */

public class StartRepluginManager {
    private static StartRepluginManager instance;
    protected StartRepluginManager(){

    }
    public static StartRepluginManager getInstance(){
        if (instance == null) {
            instance = new StartRepluginManager();
        }
        return instance;
    }
    public void startRepluginActivity(Activity activity, Intent intent, int requestCode){

    }
}

在Base的ActivityStack的路由方法里包含这样一段内容。稍微有点基础都看得懂是怎么回事,就是首先初始化StartRepluginManager的单例对象,然后调用这个类里面的startRepluginActivity方法。我们看到上面这个方法实际上是个空的,也就是这个调用其实没什么效果。

{
        /**
         * 待实现
         */
        if (startRepluginManagerInstance == null) {
            startRepluginManagerInstance = getStartRepluginManagerInstance();
        }

        startRepluginManagerInstance.startRepluginActivity(activity, intent, requestCode);
}
    private StartRepluginManager getStartRepluginManagerInstance() {

        return StartRepluginManager.getInstance();
    }

我们思路是这样的,hook节点一般是单例对象或静态变量,我们选择StartRepluginManager的单例对象作为hook节点,在Host里面在定义一个StartRepluginManager,里面的startRepluginActivity方法跟Base里面的方法不一样,不再是空的方法了。然后用新的StartRepluginManager替换之前的,这样就实现了偷梁换柱。

2. 在Host的Application里进行hook注入

 private void handleActivityStackHook() {
        try {
            //添加ActivityStackHost
            ActivityStack activityStack = ActivityStack.getInstance();
            Class<?> activityStackClass = Class.forName("com.xxx.core.ActivityStack");
            Field startRepluginManagerInstance = activityStackClass.getDeclaredField("startRepluginManagerInstance");
            startRepluginManagerInstance.setAccessible(true);
            StartRepluginManagerHook startRepluginManagerHook =  StartRepluginManagerHook.getInstance();
            startRepluginManagerInstance.set(activityStack, startRepluginManagerHook);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }  catch (IllegalAccessException e) {
            e.printStackTrace();
        }  catch (NoSuchFieldException e) {
            e.printStackTrace();
        }
    }

以上就是Hook全部代码,解释一下:
- 核心是拿到ActivityStack的成员变量startRepluginManagerInstance,也就是StartRepluginManager的单例对象。
- 然后创建新的Host里的StartRepluginManagerHook单例对象。
- startRepluginManagerInstance.set就是将新的单例对象设置给activityStack对象。
最后贴一下Host里新的StartRepluginManagerHook类代码,注意到方法里是有内容的。

/**
 * Created by zhouzhidong on 2017/10/24.
 * 开启插件的Activity的hook类
 */

public class StartRepluginManagerHook extends StartRepluginManager{
    private static StartRepluginManagerHook instance;

    protected StartRepluginManagerHook() {

    }

    public static StartRepluginManagerHook getInstance(){

        if (instance == null) {
            instance = new StartRepluginManagerHook();
        }
        return instance;
    }
    public void startRepluginActivity(Activity activity, Intent intent, int requestCode){
        RePlugin.startActivityForResult(activity, intent, requestCode);
    }
}

运行结果

我们发现,在Base里调用StartRepluginManager 的startRepluginActivity方法时,会走Host里的StartRepluginManager 的该方法,执行RePlugin.startActivityForResult(activity, intent, requestCode);这行代码。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值