Kotlin 动态代理的使用以及一些坑点

Kotlin 动态代理的使用以及一些坑点

什么是动态代理

在本文中,讨论的是kotlin中如何使用类似于JDK动态代理的功能,而非kotlin中by关键字实现的代理模式,所以可以说此文是介绍如何用kotlin实现JDK动态代理,至于JDK动态代理是什么,本文不予赘述,读者可自行查阅有关资料。

如何使用

在正式开始使用前,先介绍下几个比较重要的东西

  • 在动态代理中,最重要的是InvocationHandler,因为最终实现代理就是在InvocationHandlerinvoke方法中实现的,InvocationHandler是一个接口,该接口中只有一个invoke方法。
  • 要被代理的类必须继承自一个接口,换句话说,要被代理的方法,必须利用接口进行声明。

下面开始正片:

  1. 为需要被代理的类及方法创建接口
    interface IDemo{
        fun test(action: String, code: Int)
    }
    
  2. 创建实现接口的类
    class Demo : IDemo{
        override fun test(action: String, code: Int) {
            println("action = [${action}], code = [${code}]")
        }
    }
    
  3. 进行代理
    fun proxyTest() {
        val demo = Demo() // 创建要被代理的实例
        val proxy = Proxy.newProxyInstance( // 获取实例的代理对象
            Demo::class.java.classLoader, // 获取实例的classloader
            arrayOf(IDemo::class.java), // 实例接口
            object : InvocationHandler {
                override fun invoke(proxy: Any, method: Method, args: Array<out Any>): Any {
                    println("before...") // 代理增强方法
                    val result = method.invoke(demo, *args) // 反射调用实例的原始方法
                    if ("void" == method.genericReturnType.typeName) // 如果返回值为void, 要转换为Unit, 否则会报空指针异常
                        return Unit
                    return result
                }
            }) as IDemo // 类型转换
        proxy.test("hello", 123456) // 调用代理
    }
    
  4. 输出
    before...
    action = [hello], code = [123456]
    

坑点

  1. kotlin直接生成的InvocationHandler,其invoke方法参数类型都是可空的,这里需要手动调整为非空,就是去掉参数类型声明后的?,一般来说这么做是没有问题的,可以参照JDK的InvocationHandler,当然你也可以进行非空判断,至于kotlin为什么将invoke方法的形参类型都变成了可空类型,我也不太清楚,有了解的朋友可以评论区科普一下。
  2. 返回类型,如果被代理方法的返回类型为void,那么需要在invoke方法中返回Unit
  3. ProxynewProxyInstance第二个参数必须为接口的class数组,不能是被代理对象的class,否则会报is not interface的异常。

最后,这篇文章属于个人经验,可能存在盲点,欢迎各位指出文中的错误之处!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Kotlin 的 Fragment 中使用动态广播可以按照以下步骤进行: 1. 注册广播接收器 在 Fragment 中,可以通过 `requireActivity()` 方法获取到所在的 Activity,然后调用 `registerReceiver()` 方法注册广播接收器。如下所示: ```kotlin val receiver = object : BroadcastReceiver() { override fun onReceive(context: Context?, intent: Intent?) { // 处理接收到的广播 } } val filter = IntentFilter("com.example.ACTION") requireActivity().registerReceiver(receiver, filter) ``` 上述代码中,我们创建了一个名为 `receiver` 的匿名内部类,继承自 `BroadcastReceiver` 类,并实现了 `onReceive()` 方法来处理接收到的广播。然后,创建了一个名为 `filter` 的 `IntentFilter` 对象,用于指定所要接收的广播类型。最后,调用 `requireActivity().registerReceiver(receiver, filter)` 方法注册广播接收器。 2. 取消广播接收器 在 Fragment 销毁时,需要取消广播接收器的注册,以避免内存泄漏。可以在 `onDestroy()` 方法中调用 `requireActivity().unregisterReceiver(receiver)` 方法取消注册。完整代码如下所示: ```kotlin class MyFragment : Fragment() { private val receiver = object : BroadcastReceiver() { override fun onReceive(context: Context?, intent: Intent?) { // 处理接收到的广播 } } override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { // 创建视图 } override fun onResume() { super.onResume() val filter = IntentFilter("com.example.ACTION") requireActivity().registerReceiver(receiver, filter) } override fun onDestroy() { super.onDestroy() requireActivity().unregisterReceiver(receiver) } } ``` 上述代码中,我们将广播接收器的创建、注册和取消注册分别放在了 `onCreateView()`、`onResume()` 和 `onDestroy()` 方法中。这样做可以确保广播接收器的生命周期与 Fragment 的生命周期保持一致。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值