Android下hook点击事件。(API14以上)

最近需要在现有的app中设置统计埋点。去业务代码里埋的话似乎耦合度太高。所以决定使用hook的方法对事件进行埋点处理。
这里先记一下对点击事件hook的基本流程。

1.先建一个代理类实现View.OnClickListener,用来做点击后的后续处理。
import android.view.View;

/**
 * 实现点击监听
 */
public class OnClickListenerProxy implements View.OnClickListener{
    private View.OnClickListener mOriginalListener;

    //直接在构造函数中传进来原来的OnClickListener
    public OnClickListenerProxy(View.OnClickListener originalListener) {
        mOriginalListener = originalListener;
    }

    @Override public void onClick(View v) {
        if (mOriginalListener != null) {
            mOriginalListener.onClick(v);
        }
        Log.d("LOGCAT","hooked!");
    }
}
2.通过java的反射机制进行hook
public static void hookOnClickListener(View view) {
        try {
            // 得到 View 的 ListenerInfo 对象
            Method getListenerInfo = View.class.getDeclaredMethod("getListenerInfo");
            //修改getListenerInfo为可访问(View中的getListenerInfo不是public)
            getListenerInfo.setAccessible(true);
            Object listenerInfo = getListenerInfo.invoke(view);
            // 得到 原始的 OnClickListener 对象
            Class<?> listenerInfoClz = Class.forName("android.view.View$ListenerInfo");
            Field mOnClickListener = listenerInfoClz.getDeclaredField("mOnClickListener");
            mOnClickListener.setAccessible(true);
            View.OnClickListener originOnClickListener = (View.OnClickListener) mOnClickListener.get(listenerInfo);
            // 用自定义的 OnClickListener 替换原始的 OnClickListener
            View.OnClickListener hookedOnClickListener = new OnClickListenerProxy(originOnClickListener);
            mOnClickListener.set(listenerInfo, hookedOnClickListener);
        } catch (Exception e) {
            Log.d("LOGCAT","hook clickListener failed!", e);
        }
    }
3.在你需要hook的事件后调用上面这个hookOnClickListener
        Button btnSend = (Button) findViewById(R.id.btn_send);
        btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                log.info("onClick");
            }
        });
        HookManager.hookOnClickListener(btnSend);
4.作为统计埋点,不免需要带点参数

在原业务代码的onClick里设置参数

private View.OnClickListener clickBtn = new Button.OnClickListener(){
        @Override
        public void onClick(View v) {
            Map map = new HashMap();
            map.put("name",v.getClass().getName());
            v.setTag(v.getId(),map);
            HookManager.hookOnClickListener(v);
        }
    };

在自定义的代理onClick里接收参数

@Override public void onClick(View v) {
        if (mOriginalListener != null) {
            mOriginalListener.onClick(v);
        }
//        Log.d("LOGCAT","hooked!"+v.getId());
        //拿到之前传递的参数
        Object obj = v.getTag(v.getId());
        Log.d("LOGCAT","hooked!"+v.getId()+"_"+obj.toString());
    }
至此就可以在hook里随意加入后续操作而不用改动原来的逻辑代码了。

其实,每次在原来的onClick之后调用hook的话,耦合度还是有点高,而且hook的意义也不明显。然而如果用遍历所有view并hook事件的方法的话,统计数据的分离和处理又十分困难,不知道有没有什么两全齐美的办法。


相关github地址:https://github.com/codeqian/android-class-lib/tree/master/utilDemo/app/src/main/java/Hook

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,995评论 25 708
  • 平常的时候容易提醒自己:得意别忘了挨摔!但是真正到了得意的时候,却时常让自己“不正常”起来。
    南山青阅读 190评论 0 1
  • 巴渝墨客 2018-6-9 现在我们的生活,正在并已经失去了许多的记忆。今天是这个样子,那往后呢?往后又会...
    巴渝墨客阅读 638评论 0 0
  • 今晚偶然在朋友圈看见一同学分享的一首歌《借我》 歌词如下 然后,在网易云找到后就一直开始循环。 可能是太触动心弦。...
    一枚鸽子要飞远阅读 251评论 0 0
  • 如果一切都刚刚好 花儿开了 鸟儿叫了 充满着朝气蓬勃的生活 就连雨天也是快乐的 如果一切都刚刚的好 日月同辉 光影...
    芝初阅读 333评论 16 24