弱引用WeakReference想必不需要多说,常用来解决内存泄漏问题。它的使用应该说还算简单,但和普通变量直接存取相比,还是有那么一点繁琐的,而kotlin的委托属性主要就是解决setter和getter的,用它完全可以简化弱引用的使用。
下面直接放出自定义委托的代码,如果对委托属性不熟悉,请自行上网查找资料。
class Weak<T : Any>(initializer: () -> T?) {
var weakReference = WeakReference<T?>(initializer())
constructor():this({
null
})
operator fun getValue(thisRef: Any?, property: KProperty<*>): T? {
Log.d("Weak Delegate","-----------getValue")
return weakReference.get()
}
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T?) {
Log.d("Weak Delegate","-----------setValue")
weakReference = WeakReference(value)
}
}
代码很少,首先声明了一个泛型T,而getValue的返回值和setValue的value参数为T?,也就是说被委托的属性必须是可为null的,这个也很合理,毕竟是使用了弱引用,一旦持有的对象被回收了,获取到的自然就是null了,所以属性必须可null。剩下的代码就很简单了,类内部持有一个弱引用对象,在类创建的时候把initializer方法的返回值作为初始值存入弱引用,getValue方法从弱引用取值,setValue方法给弱引用赋值,而如果你不想为属性赋初值的话,可以使用无参构造器,其实就是给弱引用的初值为null了。
具体使用,假设act是Activity类型的
//需要指定初始值的情况
//自动推断出泛型
var act by Weak{
context
}
//也可以指定泛型,一种是给属性指定类型,必须为可null的
var act: Activity? by Weak {
context
}
//第二种是为Weak指定泛型,不可null的
var act by Weak<Activity> {
context
}
//不指定初始值的情况,此时必须指定泛型
var act:Activity? by Weak()
或者
var act by Weak<Activity>()
这样对属性赋值和取值时像普通属性那样就可以了,委托内部已经帮我们做好了弱引用。
这篇文章其实很简单,之所以还要发出来是因为我发现很多人对委托还只停留在用用lazy的阶段,希望发出来能为不熟悉委托的人带来一点帮助吧