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