我来了我来了,今天给大家介绍一下数据埋点相关知识
首先数据埋点分为三种。
一 代码埋点
这种是我接触最多的埋点方式,直接进行代码侵入埋点举个例子
viewById.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
execute();
// Log.i(TAG, "after: " + System.currentTimeMillis() -beforTime);
}
});
logi这一行就属于代码埋点
优点:灵活性高可以定制想要的属性和字段
缺点:对代码侵入,埋点必须熟悉业务的专业人员,工作量比较大,更新埋点通过发版来进行
二 全埋点
两种实现 第一种hook系统方法,比如hook所有view的onclick事件 ,然后替换成自己的onclick事件。
举个例子
public static void hookListener(Activity mActivity) {
if (null != mActivity) {
View view = mActivity.getWindow().getDecorView();
getView(view);
}
}
private static void getView(View view) {
if (view instanceof ViewGroup) {
for (int i = 0, count = ((ViewGroup) view).getChildCount(); i < count; i++) {
getView(((ViewGroup) view).getChildAt(i));
}
}
viewHook(view);
}
private static void viewHook(View view) {
try {
Class viewClass = Class.forName("android.view.View");
Field feild = viewClass.getDeclaredField("mListenerInfo");
feild.setAccessible(true);
Object mListenerInfo = feild.get(view);//view对象中的mListenerInfo实例
if (null != mListenerInfo) {
Class listenerInfo2 = Class.forName("android.view.View$ListenerInfo");//反射创建listenerinfo
Field mOnClickListener = listenerInfo2.getDeclaredField("mOnClickListener");
mOnClickListener.setAccessible(true);
View.OnClickListener lis = (View.OnClickListener) mOnClickListener.get(mListenerInfo);
if (lis != null) {
View.OnClickListener mOnClickListenerWrapper = new OnClickListenerWrapper(lis);
mOnClickListener.set(mListenerInfo,mOnClickListenerWrapper);//设置代理
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
这样hook掉所有View的点击事件,就可以监听到所有view的点击事件了,缺点由于使用反射会消耗app的性能,影响运行速度。
第二种 aop方法,简单解释一下aop,java是面向对象编程,aop面向切面编程,通俗的讲它不关心对象,他关注的是某一个点,比如咱们方法中的单击事件,这就是一个点,意思就是所有的单击事件都是我的点,我是根据点来进行编程,只要是这个点就归我管。
看一个例子
@Aspect
public class AspectJAnn {
private static final String TAG = "AspectJAnn";
@Before("execution(* android.view.View.OnClickListener.onClick(..))")
public void clickMethod(JoinPoint joinPoint) {
try {
Object[] args = joinPoint.getArgs();
for (int i = 0; i < args.length; i++) {
View pointView = (View) args[i];
String entryName = BaseApplication.getInstance().getResources().getResourceEntryName(pointView.getId());
if (pointView instanceof TextView){
entryName = ((TextView) pointView).getText().toString();
}
AnnHelper.get().onAnnClick(entryName, "no_from", ActivityCollector.getCurrentActivity().getClass().getName());
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
写了这个类,不用任何地方调用,我编译之后,所有的onclick方法都会执行到这里。(jar中包的除外,只能是我们自己写的代码,就是省去了我们自己写)
原理:通过一个插件,在编译时期(编译时期是重点)把这个方法的调用自动插入到所有onclick方法中了,这也就是jar包中的单击方法不能插入的原因。可以通过app反编译后看到具体的实现,我虽然看过但是忘记截图了,这里就不上图了
这种方案校对上一种方案的优点,编译时候加的代码,不会影响运行性能,但是会影响编译速度,不支持jar包,不支持喇嘛大表达式,而且导入包的时候我经历了各种莫名其妙绝望的错误,最后还好在各种朋友的帮助下完成了集成(朋友是我们一生的财富)。
附上一篇使用说明和github地址:使用说明,github地址
优点:没有代码侵入,工作量很小
缺点:灵活性太差,拿不到特别想要的数据。
三 可视化埋点
简单介绍,这种没有太多用武之地,说完你就明白了。
类似全埋点的基础上,根据后台返回的具体坐标 比如距上下左右都是50dp的位置。这样弊端 如果视图修改就没什么用了。也没有遇到过这种埋点
参考 神策数据 Android 埋点白皮书。