Android Hook API实例

通过一个简单的例子,学习Hook API技术在Android 的应用。
因为介绍Hook技术的文章,往往概念性多,难以理解。
从例子入手,先不管原理,把例子代码写一遍,会有意想不到的收获。
不过,需要对Java 反射机制有一定了解。

目标:我们对一Button的响应事件做一些修改, 在原来的onClick 响应的前后加上提示,不改变原理的onClick操作

一、预备知识

(1) View.setOnClickListener的源码实现

    public void setOnClickListener(@Nullable OnClickListener l) {
        if (!isClickable()) {
            setClickable(true);
        }
        getListenerInfo().mOnClickListener = l;
    }

可以看出,传入的参数l 是赋值给 ListenerInfo 对象的mOnClickListener

    @UnsupportedAppUsage
    ListenerInfo getListenerInfo() {
        if (mListenerInfo != null) {
            return mListenerInfo;
        }
        mListenerInfo = new ListenerInfo();
        return mListenerInfo;
    }

因此,我们的目标是先获取到View 的ListenerInfo 对象,然后改变它的mOnClickListener值

二、具体过程

1. 原始OnClickListener的实现

        //1.原始代码
        val clickButton = findViewById<Button>(R.id.clickButton).apply {
            setOnClickListener {
                showToast("onclick working")
            }
        }

2. 创建一个View.OnClickListener的子类,并且它组合了原始的OnClickListener对象

    //2-1.包装类 - 包装原始的 OnClickListener, 并提供额外操作
    inner class HookedOnClickListener(private val origin: View.OnClickListener) :
        View.OnClickListener {
        override fun onClick(v: View?) {
            this@MainActivity.showToast("hook click - before") //额外操作
            origin?.onClick(v)
            this@MainActivity.showToast("hook click - after") //额外操作
        }
    }

*说明:
这里可以理解为静态代理,即这是个代理类
origin 即为原始的OnClickListener 对象
origin?.onClick(v) 即调用原来的点击响应

3. 设置hook

    //3.设置hook
    private fun doHookOnClickListener(view: View) {
        try {
            // (1) 获取view的 ListenerInfo 对象 (实例对象)
            val getListenerInfo = View::class.java.getDeclaredMethod("getListenerInfo")
            getListenerInfo.isAccessible = true
            val listenerInfo = getListenerInfo.invoke(view)

            //(2) 获取原始的 OnClickListener对象
            val listenerInfoClazz = Class.forName("android.view.View\$ListenerInfo")//类对象
            val mOnClickListener: Field = listenerInfoClazz.getDeclaredField("mOnClickListener")
            mOnClickListener.isAccessible = true
            val originOnClickListener: View.OnClickListener =
                mOnClickListener.get(listenerInfo) as View.OnClickListener

            //(3) 用自定义的OnClickListener 替换原始的
            val hookedOnClickListener = HookedOnClickListener(originOnClickListener)
            mOnClickListener.set(listenerInfo, hookedOnClickListener)
        } catch (e: Exception) {
            showToast("doHookOnClickListener e:$e")
        }
    }

可以看到,这里主要是 反射的操作,需要对反射机制熟悉

4. 执行效果

依次弹出以下Toast:
"hook click - before"
"onclick working"
"hook click - after"

三、文献参考:
https://cloud.tencent.com/developer/article/1102761

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 前言 手把手讲解系列文章,是我写给各位看官,也是写给我自己的。文章可能过分详细,但是这是为了帮助到尽量多的人,毕竟...
    波澜步惊阅读 38,211评论 21 151
  • 概要 我们在之前的一篇Android-Framework-Plugin插件话框架-Hook Activity过程 ...
    AntCoding阅读 10,342评论 0 3
  • 引子 Hook技术在android开发领域算是一项黑科技,那么一个新的概念进入视线,我们最关心的3个问题就是,它是...
    程序员小花阅读 6,081评论 0 0
  • 一.简介 Hook技术是一种用于改变API执行结果的技术,Android系统中有一套自己的事件分发机制,所有的代码...
    雷涛赛文阅读 5,176评论 0 2
  • Hook是什么? Hook 又叫“钩子”,它可以在事件传送的过程中截获并监控事件的传输,将自身的代码与系统方法进行...
    奔跑吧哈哈阅读 7,647评论 0 23

友情链接更多精彩内容