Android Context Hook

转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/120954983
本文出自【赵彦军的博客】

文章目录

Context Hook

Android 编程中,我们常常会和 Context打交道,而且 Context 遍布各个地方,就算使用 Jetpack Compose 也都离不开它。正因为 Context 被广泛的使用和传播,当我们面对一些特殊问题时,常常能够从 Context对象入手,去解决许多看似不能改变的代码问题。这常常就会用到 Context Hook 这种手法。

Context Hook 形式其实特别简单,就是使用 ContextWrapper 对原有的 Context 进行代理,从而实现 Context各个 get方法的拦截。广义的说,这其实更是一种手法或者思路,所以我们不该局限于Context 对象,只是 Context 常常成为我们的目标对象。

假设我们要拦截 Context 对象的 getString 方法,鱿鱼 getString方法在 Context中是final的,所以我们不能直接在 ContextWrapper 中覆写它,而是先去覆写 getResources , 然后返回一个 ResourcesWrapper 对象,这个 ResourcesWrapper的思想和 Context 是一模一样的,只是在 Android SDK 中并没有 ResourcesWrapper 这个类,不过在 AndroidX appcompat 库中倒是有一个, 懒得自己写的话就直接去吧 androidx.appcompat.widget.ResourcesWrapper 复制一份过来。

所以我们可以这么覆写 ContextgetResouces 方法。

class HookContext(context: Context) : ContextWrapper(context) {

    private var hookResources: HookResources? = null

    override fun getResources(): Resources {
        val originalResources = super.getResources()
        if (hookResources == null) {
            hookResources = HookResources(originalResources)
        }
        return hookResources!!
    }
}

其实这么写是有问题的,我们为了避免在 getResources 中返回的创建 HookResources对象,于是将它缓存在 HookContext 内部中,但如果 Configuration 变了,比如屏幕发生旋转,那么我们缓存的 HookResources对象身上的 Configuration 并不会被自动更新,这将导致一些很难排查的问题,所以进一步的写法是:

class HookContext(context: Context) : ContextWrapper(context) {

    private var hookResources: HookResources? = null

    override fun getResources(): Resources {
        val originalResources = super.getResources()
        if (hookResources == null) {
            hookResources = HookResources(originalResources)
        }
        val result = hookResources!!
        if (result.configuration != originalResources.configuration || result.displayMetrics != originalResources.displayMetrics) {
            result.updateConfiguration(
                originalResources.configuration,
                originalResources.displayMetrics
            )
        }
        return result
    }
}

HookResources

class HookResources(private val resources: Resources) :
    Resources(resources.assets, resources.displayMetrics, resources.configuration) {

    override fun getString(id: Int): String {
        if (id == 123) {
            return "哈哈哈,这是代理返回的"
        }
        return resources.getString(id)
    }
}

这样以后,我们对这个 HookContext对象调用 getString 时,传入123 参数,它将返回 哈哈哈,这是代理返回的 , 这就是 context.getString(123) 的返回结果。

细品一下,是不是有了更多的可能性,也可以用这中手法来拦截 ContextgetDrawable 方法。

自定义 view 使用 HookContext

class MyTextView(context: Context, attrs: AttributeSet?) :
    androidx.appcompat.widget.AppCompatTextView(HookContext(context), attrs) {

    init {
        //这里用 getContext 它是被 HookContext 包装过的 。
        //不能用 context ,它没有被包装
        text = getContext().getString(123)
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值