Kotlin-委托剖析(2)- 属性委托

2022-04-01  本文已影响0人  杨0612
简单例子

以下是属性委托的代码,

class Test {
    var a: Int = 0
    var b: Int by ::a 
} 
代码反编译
public final class Test {
   private int a;
   @NotNull
   private final KMutableProperty0 b$delegate = new Test$b$2((Test)this);//1

   public final int getA() {
      return this.a;
   }

   public final void setA(int var1) {
      this.a = var1;
   }

   public final int getB() {
      KProperty0 var1 = (KProperty0)this.b$delegate;
      Object var3 = null;
      boolean var4 = false;
      return ((Number)var1.get()).intValue();//2
   }

   public final void setB(int var1) {
      KMutableProperty0 var2 = this.b$delegate;
      Object var4 = null;
      Integer var5 = var1;
      boolean var6 = false;
      var2.set(var5);//3
   }
}

看看Bb2类是何方神圣

final class B$b$2 extends MutablePropertyReference0Impl {
   B$b$2(B var1) {
      super(var1, B.class, "a", "getA()I", 0);
   }

   @Nullable
   public Object get() {
      return ((B)this.receiver).getA();//1
   }

   public void set(@Nullable Object value) {
      ((B)this.receiver).setA(((Number)value).intValue());//2
   }
}
属性委托使用场景
  1. 改变属性名称
    a是旧版本的属性名称,在考虑新旧版本兼容的情况下,不能直接把a的名称修改了,所以定义属性b运行在新版本当中,而两者的值是一致的。
class A {
    var a: Int = 0
    var b: Int by ::a //1
} 
  1. 控制属性访问权限
    希望外部只有_names 的只读权限,定义只读类型的name委托给_names 处理,相当于将name的get权限委托给了_names 。
class A {
    private val _names = mutableListOf<String>()
    val name: List<String> by ::_names
}
  1. ViewModel构建
//手动构建
val viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

//委托构建
val viewModel:MainViewModel by viewModels()

分析源码得知道,viewModel的构建委托给了ViewModelLazy来处理,其内部也是通过ViewModelStore来获取ViewModel,这点跟ViewModelProvider一致,不过
ViewModelLazy是懒加载且增加了ViewMode的缓存处理。

总结

属性委托本质就是将set、get逻辑委托给了另外一个对象进行处理。

以上分析有不对的地方,请指出,互相学习,谢谢哦!

上一篇下一篇

猜你喜欢

热点阅读