美文网首页
无痕埋点简单实现

无痕埋点简单实现

作者: 中路杀神ai | 来源:发表于2021-12-17 18:11 被阅读0次

hook + aop 简单实现

1. View的层级

1639734894(1).jpg 1.jpg

View 所有的Listener都存储在getListenerInfo()=ListenerInfo
所有hook getListenerInfo()拿到mOnClickListener代理到我们自己的WrapperOnClickListener里

1.hook setOnClickListener
   fun hookViewOnClickListener(view: View) {
        val listener = getOnClickListener(view)
        if(listener !is WrapperViewOnClickListener){
             view.setOnClickListener(WrapperViewOnClickListener(listener))
        }
 }
2.反射拿到getListenerInfo,通过ListenerInfo拿到mOnClickListener
private fun getOnClickListener(view: View): View.OnClickListener? {
        val hasOnClick = view.hasOnClickListeners()
        if (hasOnClick) {
            try {
                val viewClazz: Class<*> = View::class.java
                val method: Method = viewClazz.getDeclaredMethod("getListenerInfo")
                if (!method.isAccessible) {
                    method.isAccessible = true
                }
                val listenerInfoObj: Any = method.invoke(view)
                val listenerInfoClazz = Class.forName("android.view.View\$ListenerInfo")
                val field: Field = listenerInfoClazz.getDeclaredField("mOnClickListener")
                if (!field.isAccessible) {
                    field.isAccessible = true
                }
                return field.get(listenerInfoObj) as View.OnClickListener
            } catch (e: Exception) {
                e.printStackTrace()
            }
        }
        return null
    }
}

3.WrapperViewOnClickListener

  class WrapperViewOnClickListener(val listener: View.OnClickListener?) :   View.OnClickListener {
    override fun onClick(v: View?) {
        Log.e("TAG","自动埋点")
        listener?.onClick(v)
        val path = ViewPath.getPath(v, StringBuilder())
        Log.e("TAG", path)
    }
}

获取当前View所在Activity的路径

注意RecyclerView 子item的位置索引
fun getPath(view: View?, path: StringBuilder): String? {
            if (view != null) {
                if (path.isEmpty()) path.insert(0, view.javaClass.simpleName)
            } else {
                return ""
            }
            if (view.parent is ViewGroup) {
                if (view.parent is RecyclerView || view.parent is AdapterView<*>) {
                    val listView: View = view.parent as View
                    var index = 0
                    if (view.parent is RecyclerView) {
                        index = (view.parent as RecyclerView).getChildAdapterPosition(view)
                    } else if (view.parent is AdapterView<*>) {
                        index = (view.parent as AdapterView<*>).getPositionForView(view)
                    }
                  path.insert(0, view.parent.javaClass.simpleName + "[" + index + "]|")
                } else {
                   path.insert(0, view.parent.javaClass.simpleName + "[" + (view.parent as ViewGroup).indexOfChild(view) + "]|")
                }
            }
            //遇到PhoneWindow退出递归
            return if (view.parent is View) {
                //不统计统计DecorView遇到Decorview退出递归 
                if ((view.parent as View).id === android.R.id.content) {
                    path.insert(0, view.parent.javaClass.simpleName + "|")
                    path.toString()
                } else {
                    getPath(view.parent as View, path)
                }
            } else {
                path.toString()
            }
        }

效果:

TAG: 自动埋点
TAG: DecorView[0]|LinearLayout[1]|FrameLayout[0]|FitWindowsLinearLayout[1]|ContentFrameLayout[0]|LinearLayout[4]|FrameLayout[0]|FrameLayout[0]|FrameLayout[0]|FrameLayout[0]|FrameLayout[0]|FrameLayout[0]|FrameLayout[0]|AppCompatButton
or
ContentFrameLayout|ContentFrameLayout[0]|LinearLayout[4]|AppCompatButton

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <FrameLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content">
        <FrameLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content">
            <FrameLayout
                android:layout_width="wrap_content"
                android:layout_height="wrap_content">
                <FrameLayout
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content">
                    <FrameLayout
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content">
                        <FrameLayout
                            android:layout_width="wrap_content"
                            android:layout_height="wrap_content">
                            <FrameLayout
                                android:layout_width="wrap_content"
                                android:layout_height="wrap_content">
                                <Button
                                    android:id="@+id/get"
                                    android:layout_width="wrap_content"
                                    android:layout_height="wrap_content"
                                    android:text="埋点" />
                            </FrameLayout>
                        </FrameLayout>
                    </FrameLayout>
                </FrameLayout>
            </FrameLayout>
        </FrameLayout>
    </FrameLayout>

</LinearLayout>

相关文章

  • 无痕埋点简单实现

    hook + aop 简单实现 1. View的层级 View 所有的Listener都存储在getListene...

  • iOS无痕埋点方案分享探究

    iOS无痕埋点方案分享探究 iOS无痕埋点方案分享探究

  • 无痕埋点

    一、概念 通过技术手段无差别地记录用户在前端页面上的行为。可以正确的获取 PV、UV、IP、Action、Time...

  • web 埋点

    数据埋点是什么?设置数据埋点的意义?web 埋点实现原理了解一下 前端监控和前端埋点方案设计美团点评前端无痕埋点实践

  • ios 无痕埋点,两种方式

    FNKTrack 两种无痕埋点方式 FNKAopTrack 用Aspects进行AOP无痕埋点,具体见代码. FN...

  • iOS埋点(无痕埋点的设计与实现)

    在移动互联网时代,对于每个公司、企业来说,用户的行为数据非常重要。重要到什么程度,用户在这个页面停留多久、点击了什...

  • 无痕埋点实践

    参考了一些实现方案,自己选择了一种适合自己项目的,这里我将进行总结。 主体思路 主要思路就是要通过方法交换实现。 ...

  • ASM 无痕埋点

    ASM 是一个 Java 字节码操控框架。它能被用来动态生成类或者增强既有类的功能。ASM 可以直接产生二进制 c...

  • Growingio

    Growingio 数据统计,无痕埋点技术;低耦合,简单易用地址 https://www.growingio.co...

  • Asm初探

    最近项目中产品要求接入神策埋点,神策最大的宣传点应该就是所谓无痕全埋点。对于这种"无痕"或者"无感知",大部分An...

网友评论

      本文标题:无痕埋点简单实现

      本文链接:https://www.haomeiwen.com/subject/ujwifrtx.html