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

简单例子

以下是属性委托的代码,

class Test {
    var a: Int = 0
    var b: Int by ::a 
} 
  • 属性b委托给了属性a,a、b两者数据同步。
    Tips:可空属性好像不能成为委托对象,例如 var a: Int? = null,就无法编译通过;
代码反编译
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
   }
}
  • 注释1:定义了KMutableProperty0类型的对象b$delegate,看名字就知道它跟委托有关;
  • 注释2:b属性的get函数其实是转移给b$delegate对象进行处理;
  • 注释3:b属性的set函数其实是转移给b$delegate对象进行处理;

看看B2类是何方神圣

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:外部调用getB()其实是调到get(),这里就是获取a属性值;
  • 注释2:外部调用setB()其实是调到set(),这里就是给a属性赋值;
属性委托使用场景
  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()
    val name: List 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逻辑委托给了另外一个对象进行处理。

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

你可能感兴趣的:(Kotlin-委托剖析(2)- 属性委托)