委托属性 是一种通过委托实现拥有 getter 和可选 setter 的 属性,并允许实现可复用的自定义属性。例如:
class Example {
var p: String by Delegate()
}
委托对象必须实现一个拥有 getValue() 方法的操作符,以及 setValue() 方法来实现读/写属性。些方法将会接受包含对象实例以及属性元数据作为额外参数。当一个类声明委托属性时,编译器生成的代码会和如下 Java 代码相似。
public final class Example {
@NotNull
private final Delegate p$delegate = new Delegate();
// $FF: synthetic field
static final KProperty[] $$delegatedProperties = new KProperty[]{(KProperty)Reflection.mutableProperty1(new MutablePropertyReference1Impl(Reflection.getOrCreateKotlinClass(Example.class), "p", "getP()Ljava/lang/String;"))};
@NotNull
public final String getP() {
return this.p$delegate.getValue(this, $$delegatedProperties[0]);
}
public final void setP(@NotNull String var1) {
Intrinsics.checkParameterIsNotNull(var1, "");
this.p$delegate.setValue(this, $$delegatedProperties[0], var1);
}
}
一些静态属性元数据被加入到类中,委托在类的构造函数中初始化,并在每次读写属性时调用。
委托实例在上面的例子中,创建了一个新的委托实例来实现属性。这就要求委托的实现是有状态的,例如,当其内部缓存计算结果时:
class StringDelegate {
private var cache: String? = null
operator fun getValue(thisRef: Any?, property: KProperty): String {
var result = cache
if (result == null) {
result = someOperation()
cache = result
}
return result
}
}
与此同时,当需要额外的参数时,需要建立新的委托实例,并将其传递到构造器中。
class Example {
private val nameView by BindViewDelegate(R.id.name)
}
但也有一些情况是只需要一个委托实例来实现任何属性的:当委托是无状态,并且它所需要的唯一变量就是已经提供好的包含对象实例和委托名称时,可以通过将其声明为 object 来替代 class 实现一个单例委托。
举个例子,下面的单例委托从 Android Activity 中取回与给定 tag 相匹配的 Fragment。
object FragmentDelegate {
operator fun getValue(thisRef: Activity, property: KProperty): Fragment? {
return thisRef.fragmentManager.findFragmentByTag(property.name)
}
}
类似地,任何已有类都可以通过扩展变成委托。getValue() 和 setValue() 也可以被声明成 扩展方法 来实现。Kotlin 已经提供了内置的扩展方法来允许将 Map and MutableMap 实例用作委托,属性名作为其中的键。 如果你选择复用相同的局部委托实例来在一个类中实现多属性,你需要在构造函数中初始化实例。 注意:从 Kotlin 1.1 开始,也可以声明 方法局部变量声明为委托属性。在这种情况下,委托可以直到该变量在方法内部声明的时候才去初始化,而不必在构造函数中就执行初始化。
泛型委托委托方法也可以被声明成泛型的,这样一来不同类型的属性就可以复用同一个委托类了。
private var maxDelay: Long by SharedPreferencesDelegate()
然而,如果像上例那样对基本类型使用泛型委托的话,即便声明的基本类型非空,也会在每次读写属性的时候触发装箱和拆箱的操作。
说明:对于非空基本类型的委托属性来说,最好使用给定类型的特定委托类而不是泛型委托来避免每次访问属性时增加装箱的额外开销。
标准委托:lazy()针对常见情形,Kotlin 提供了一些标准委托,如 Delegates.notNull()、 Delegates.observable() 和 lazy()。 lazy() 是一个在第一次读取时通过给定的 lambda 值来计算属性的初值,并返回只读属性的委托。
private val dateFormat: DateFormat by lazy {
SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
}
这是一种简洁的延迟高消耗的初始化至其真正需要时的方式,在保留代码可读性的同时提升了性能。
需要注意的是,lazy() 并不是内联函数,传入的 lambda 参数也会被编译成一个额外的 Function 类,并且不会被内联到返回的委托对象中。 经常被忽略的一点是 lazy() 有可选的 mode 参数 来决定应该返回 3 种委托的哪一种:
public fun lazy(initializer: () -> T): Lazy = SynchronizedLazyImpl(initializer)
public fun lazy(mode: LazyThreadSafetyMode, initializer: () -> T): Lazy =
when (mode) {
LazyThreadSafetyMode.SYNCHRONIZED -> SynchronizedLazyImpl(initializer)
LazyThreadSafetyMode.PUBLICATION -> SafePublicationLazyImpl(initializer)
LazyThreadSafetyMode.NONE -> UnsafeLazyImpl(initializer)
}
默认模式 LazyThreadSafetyMode.SYNCHRONIZED 将提供相对耗费昂贵的 双重检查锁 来保证一旦属性可以从多线程读取时初始化块可以安全地执行。 如果你确信属性只会在单线程(如主线程)被访问,那么可以选择 LazyThreadSafetyMode.NONE 来代替,从而避免使用锁的额外开销。
val dateFormat: DateFormat by lazy(LazyThreadSafetyMode.NONE) {
SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
}
区间
区间 是 Kotlin 中用来代表一个有限的值集合的特殊表达式。值可以是任何 Comparable 类型。 这些表达式的形式都是创建声明了 ClosedRange 接口的方法。创建区间的主要方法是 .. 操作符方法。
包含区间表达式的主要作用是使用 in 和 !in 操作符实现包含和不包含。
if (i in 1..10) {
println(i)
}
该实现针对非空基本类型的区间(包括 Int、Long、Byte、Short、Float、Double 以及 Char 的值)实现了优化,所以上面的代码可以被优化成这样:
if(1 "Find it somewhere else"
else -> "Oops"
}
相比一系列的 if{…} else if{…} 代码块,这段代码在不降低效率的同时提高了代码的可读性。然而,如果在声明和使用之间有至少一次间接调用的话,range 会有一些微小的额外开销。比如下面的代码:
private val myRange get() = 1..10
fun rangeTest(i: Int) {
if (i in myRange) {
println(i)
}
}
在编译后会创建一个额外的 IntRange 对象:
private final IntRange getMyRange() {
return new IntRange(1, 10);
}
public final void rangeTest(int i) {
if(this.getMyRange().contains(i)) {
System.out.println(i);
}
}
将属性的 getter 声明为 inline 的方法也无法避免这个对象的创建。这是 Kotlin 1.1 编译器可以优化的一个点。至少通过这些特定的区间类避免了装箱操作。
说明:尽量在使用时直接声明非空基本类型的区间,不要间接调用,来避免额外区间类的创建。或者直接声明为常量来复用。
区间也可以用于其他实现了 Comparable 的非基本类型。
if (name in "Alfred".."Alicia") {
println(name)
}
在这种情况下,最终实现并不会优化,而且总是会创建一个 ClosedRange 对象,如下面编译后的代码所示:
if(RangesKt.rangeTo((Comparable)"Alfred", (Comparable)"Alicia")
.contains((Comparable)name)) {
System.out.println(name);
}
迭代:for 循环
整型区间 (除了 Float 和 Double之外其他的基本类型)也是 级数:它们可以被迭代。这就可以将经典 Java 的 for 循环用一个更短的表达式替代。
for (i in 1..10) {
println(i)
}
经过编译器优化后的代码实现了零额外开销:
int i = 1;
byte var3 = 10;
if(i = var3) {
while(true) {
System.out.println(i);
if(i == var3) {
break;
}
--i;
}
}
然而,其他迭代器参数并没有如此好的优化。反向迭代还有一种结果相同的方式,使用 reversed() 方法结合区间:
for (i in (1..10).reversed()) {
println(i)
}
编译后的代码并没有看起来那么少:
IntProgression var10000 = RangesKt.reversed((IntProgression)(new IntRange(1, 10)));
int i = var10000.getFirst();
int var3 = var10000.getLast();
int var4 = var10000.getStep();
if(var4 > 0) {
if(i > var3) {
return;
}
} else if(i < var3) {
return;
}
while(true) {
System.out.println(i);
if(i == var3) {
return;
}
i += var4;
}
会创建一个临时的 IntRange 对象来代表区间,然后创建另一个 IntProgression 对象来反转前者的值。 事实上,任何结合不止一个方法来创建递进都会生成类似的至少创建两个微小递进对象的代码。 这个规则也适用于使用 step() 中缀方法来操作递进的步骤,即使只有一步:
for (i in 1..10 step 2) {
println(i)
}
一个次要提示,当生成的代码读取 IntProgression 的 last 属性时会通过对边界和步长的小小计算来决定准确的最后值。在上面的代码中,最终值是 9。
最后,until() 中缀函数对于迭代也很有用,该函数(执行结果)不包含最大值。
for (i in 0 until size) {
println(i)
}
遗憾的是,编译器并没有针对这个经典的包含区间围优化,迭代器依然会创建区间对象:
IntRange var10000 = RangesKt.until(0, size);
int i = var10000.getFirst();
int var1 = var10000.getLast();
if(i
关注
打赏