Android技术知识Android开发Android开发经验谈

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

2018-08-27  本文已影响119人  书柜里的松鼠

最近需要在现有的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

上一篇 下一篇

猜你喜欢

热点阅读