[译]探索Kotlin中隐藏的性能开销-Part 3

深藏阁楼爱情的钟 2023-06-10 13:23 116阅读 0赞

[译]探索Kotlin中隐藏的性能开销-Part 3

翻译说明:

原标题# Exploring Kotlin’s hidden costs — Part 3

原文地址: https://medium.com/@BladeCoder/exploring-kotlins-hidden-costs-part-2-324a4a50b70

原文作者: Christophe Beyls

代理属性和Range

在发布有关Kotlin编程语言的性能开销系列的前两篇文章之后,我收到了很多不错的反馈,甚至还包括 Jake Wharton 大神他自己。所以你还没看前两篇文章,千万不要错过哦。

在第3部分中,我们将揭开更多有关Kotlin编译器的秘密,并提供如何编写更高效代码的新技巧。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-tHlu5rBL-1572187598247)(https://miro.medium.com/max/646/1\*-iKupZ7diZBEzTw87Bkaxg.jpeg)\]

一、代理属性

代理属性是一种其getter和可选的setter的内部实现可由代理的外部对象提供的属性。它可以允许复用自定义属性的内部实现。

  1. class Example {
  2. var p: String by Delegate()
  3. }

这个代理对象必须实现一个 operator getVlue()函数,以及一个 setValue()函数来用于属性的读/写. 这些函数将接收包含对象实例 以及属性的metadata元数据 作为额外参数(比如它的属性名)。

当类中声明一个代理属性时,编译将生成以下代码(下面是反编译后的Java代码):

  1. public final class Example {
  2. @NotNull
  3. private final Delegate p$delegate = new Delegate();
  4. // $FF: synthetic field
  5. static final KProperty[] $$delegatedProperties = new KProperty[]{
  6. (KProperty)Reflection.mutableProperty1(new MutablePropertyReference1Impl(Reflection.getOrCreateKotlinClass(Example.class), "p", "getP()Ljava/lang/String;"))};
  7. @NotNull
  8. public final String getP() {
  9. return this.p$delegate.getValue(this, $$delegatedProperties[0]);
  10. }
  11. public final void setP(@NotNull String var1) {
  12. Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
  13. this.p$delegate.setValue(this, $$delegatedProperties[0], var1);
  14. }
  15. }

一些静态属性metadata元数据被添加到类中。代理将在类的构造器中进行初始化,然后在每次读取或写入属性时都调用该代理。

代理实例

在上述例子中,将会创建一个新的代理对象的实例来实现该属性。当代理实例是有状态的时候, 这就是必需的,例如在计算本地缓存属性的值时.

  1. class StringDelegate {
  2. private var cache: String? = null
  3. operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
  4. var result = cache
  5. if (result == null) {
  6. result = someOperation()
  7. cache = result
  8. }
  9. return result
  10. }
  11. }

如果还需要通过其构造函数传递的额外参数,则还需要创建一个新的代理实例:

  1. class Example {
  2. private val nameView by BindViewDelegate<TextView>(R.id.name)
  3. }

但是在某些情况下,只需要一个代理实例就可以实现任意属性: 当代理实例是无状态的时候,并且它执行所需的唯一变量就是对象实例和属性名称(然而这些编译器都直接提供了)。在这种情况下,可以通过将代理实例声明成object对象表达式而不是一个来使得成为单例

例如,下面的代理单例实例检索其标记名称与Android Activity 中的属性名称来匹配Fragment.

  1. object FragmentDelegate {
  2. operator fun getValue(thisRef: Activity, property: KProperty<*>): Fragment? {
  3. return thisRef.fragmentManager.findFragmentByTag(property.name)
  4. }
  5. }

同样,任意的对象都可以扩展成代理。此外getValue()setValue()还可以声明成扩展函数。Kotlin中已经提供了内置的扩展函数,例如允许将MapMutableMap实例作为代理实例,并将属性的名称作为key.

如果你选择在同一个类中实现多个属性复用同一个局部代理实例的话,那么需要在类的构造器中初始化此实例。

注意: 从Kotlin1.1开始,也可以在函数中声明局部变量作为代理属性。那么在这种情况下,代理实例可以延迟初始化,直到在函数中声明变量为止。

在类中声明的每个代理属性都涉及到其关联的代理对象创建的性能开销,并向该类中添加一些metadata元数据。必要的时候,可以尝试为不同属性复用同一个代理实例。在你声明大量代理属性的时候,还需要考虑代理属性是否你的最佳选择。

泛型代理

还可以以泛型的方式声明代理函数,因此同一个代理类可以用任意的属性类型。

  1. private var maxDelay: Long by SharedPreferencesDelegate<Long>()

但是,如果像上面例子那样使用具有原生类型属性的泛型代理的话,即便声明的原生类型为非null,每次读取或写入该属性时都避免不了装箱和拆箱的发生

对于非null原生类型的代理属性,最好使用为该特定值类型创建特定的代理类,而不是泛型代理,以避免在每次访问该属性时产生的装箱开销

标准库代理: lazy()

Kotlin内置了一些标准库代理函数来覆盖常见的情况,例如 Delegates.notNull(),Delegates.observable()lazy().

lazy(initializer: () -> T) 是一个为只读属性返回代理对象的函数,该属性是通过在其首次被读取的时,lazy函数参数lambda initializer执行来初始化的。

  1. private val dateFormat: DateFormat by lazy {
  2. SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
  3. }

这是一种将昂贵的初始化操作延迟到实际需要使用之前的巧妙方法,可以在保持代码可读性的同时又提高了性能。

需要注意到的是,lazy()函数不是内联函数,并且作为参数传递的lambda将编译成独立的Function类,并且不会在返回的代理对象内进行内联。

通常会被人忽略的是lazy()另一重载函数实际上还隐藏一个可选的模式参数来确定应该返回3种不同类型的代理中的一种:

  1. public fun <T> lazy(initializer: () -> T): Lazy<T> = SynchronizedLazyImpl(initializer)
  2. public fun <T> lazy(mode: LazyThreadSafetyMode, initializer: () -> T): Lazy<T> =
  3. when (mode) {
  4. LazyThreadSafetyMode.SYNCHRONIZED -> SynchronizedLazyImpl(initializer)
  5. LazyThreadSafetyMode.PUBLICATION -> SafePublicationLazyImpl(initializer)
  6. LazyThreadSafetyMode.NONE -> UnsafeLazyImpl(initializer)
  7. }

默认的模式是 LazyThreadSafetyMode.SYNCHRONIZED 将执行相对开销昂贵的双重锁的检查,这是为了保证在多线程环境下读取属性时,初始化块可以安全运行。

如果你明确知道当前环境是单线程(例如主线程)访问属性,那么可以通过显式使用 LazyThreadSafetyMode.NONE 来完全避免双重锁的检查所带来昂贵的开销。

  1. val dateFormat: DateFormat by lazy(LazyThreadSafetyMode.NONE) {
  2. SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
  3. }

使用lazy()代理可以按需延迟昂贵的初始化,此外可以指定线程安全的模式以避免不必要的双重锁检查。

二、Ranges(区间)

区间是一种用于表示Kotlin中的一组有限值的特殊表达式。这些值可以是任意Comparable类型。这些表达式由创建用于实现ClosedRange对象的函数形成。用于创建区间的主要函数是 ..操作符。

区间包含的测试

区间表达式主要目的是使用in!in 运算符来判断是否包含某个值

  1. if (i in 1..10) {
  2. println(i)
  3. }

该实现特地针对非null原生类型区间(有: Int, Long, Byte, Short, Float, Double或Char)进行了优化,因此上面例子可以高效编译成如下形式:

  1. if(1 <= i && i <= 10) {
  2. System.out.println(i);
  3. }

性能开销几乎为0,没有额外的对象分配。区间也可以和任意其他非原生Comparable类型一起使用。

  1. if (name in "Alfred".."Alicia") {
  2. println(name)
  3. }

在Kotlin 1.1.50之前,编译以上示例时始终会创建一个临时的ClosedRange对象。但是从1.1.50之后,已经对它的实现进行了优化,以避免Comparable类型额外开销分配:

  1. if(name.compareTo("Alfred") >= 0) {
  2. if(name.compareTo("Alicia") <= 0) {
  3. System.out.println(name);
  4. }
  5. }

此外,区间检查还包括应用再 when 表达式中

  1. val message = when (statusCode) {
  2. in 200..299 -> "OK"
  3. in 300..399 -> "Find it somewhere else"
  4. else -> "Oops"
  5. }

这使代码比一系列if {...} else if {...}语句更具可读性,并且效率更高。

但是,在区间包含检查中,当区间的声明之间至少存在一个间接过程时,会有一个小的性能开销。 比如下面这段Kotlin代码:

  1. private val myRange get() = 1..10
  2. fun rangeTest(i: Int) {
  3. if (i in myRange) {
  4. println(i)
  5. }
  6. }

上述代码会造成在编译后额外创建一个IntRange对象:

  1. private final IntRange getMyRange() {
  2. return new IntRange(1, 10);
  3. }
  4. public final void rangeTest(int i) {
  5. if(this.getMyRange().contains(i)) {
  6. System.out.println(i);
  7. }
  8. }

即使将属性getter声明成内联函数也不能避免创建IntRange对象。在这种情况下,Kotlin 1.1编译器已经改进了。 由于这些特定的区间类存在,至少在比较原生类型时不会出现装箱过程。

尝试在没有间接声明过程区间检查中使用直接声明区间的方式,来避免额外区间对象的创建分配,另外,可以将它们声明成常量以此来复用他们。

迭代: for循环

整数类型区间(除Float或Double之外的任何原生类型的区间)也是级数: 可以对其进行迭代。这允许用较短的语法替换经典的Java for循环。

  1. for (i in 1..10) {
  2. println(i)
  3. }

这可以以零开销方式编译为可比较的优化代码:

  1. int i = 1;
  2. for(byte var2 = 11; i < var2; ++i) {
  3. System.out.println(i);
  4. }

如果向后迭代,请使用 downTo() 中缀函数来替代

  1. for (i in 10 downTo 1) {
  2. println(i)
  3. }

同样,使用此构造进行编译后的开销为零:

  1. int i = 10;
  2. byte var1 = 1;
  3. while(true) {
  4. System.out.println(i);
  5. if(i == var1) {
  6. return;
  7. }
  8. --i;
  9. }

还有一个有用的until()中缀函数可以迭代直到但不包括区间上限值。

  1. for (i in 0 until size) {
  2. println(i)
  3. }

当本文的原始版本发布时,调用此函数用于生成次优代码。自Kotlin 1.1.4起,情况已大大改善,并且编译器现在生成等效的Java for循环:

  1. int i = 0;
  2. for(int var2 = size; i < var2; ++i) {
  3. System.out.println(i);
  4. }

但是,其他迭代变体的优化效果也不佳

这是另一种使用reversed() 函数与区间组合的方法,可以向后迭代并产生与downTo()完全相同的结果。

  1. for (i in (1..10).reversed()) {
  2. println(i)
  3. }

不幸的是,生成的编译代码就不那么漂亮:

  1. IntProgression var10000 = RangesKt.reversed((IntProgression)(new IntRange(1, 10)));
  2. int i = var10000.getFirst();
  3. int var3 = var10000.getLast();
  4. int var4 = var10000.getStep();
  5. if(var4 > 0) {
  6. if(i > var3) {
  7. return;
  8. }
  9. } else if(i < var3) {
  10. return;
  11. }
  12. while(true) {
  13. System.out.println(i);
  14. if(i == var3) {
  15. return;
  16. }
  17. i += var4;
  18. }

将会创建一个临时的IntRange对象来表示区间,然后再创建另一个IntProgression对象来反转第一个对象的值。

事实上,创建一个progression的以上功能任何组合都会生成类似的代码,涉及到创建至少两个轻量级progression对象的小开销。

此规则也适用于使用step()中缀函数来修改progression, 即使步长是1:

  1. for (i in 1..10 step 2) {
  2. println(i)
  3. }

附带说明下,当生成的代码读取IntProgression的最后一个属性时,这将执行少量计算,以通过考虑边界和步长来确定区间的确切最后一个值。在上面的示例中,最后一个值应该为9。

若要在for循环中进行迭代,最好使用区间表达式,该区间表达式只涉及到对 ..downTo()untill()的单个函数调用,以避免创建临时progression对象的开销。

迭代: for-each()

与其使用for循环,不如尝试在区间上使用forEach()内联扩展函数来达到相同的结果。

  1. (1..10).forEach {
  2. println(it)
  3. }

但是,如果您仔细查看此处使用的forEach()函数的签名,你会注意到,它并没有针对区间进行优化,而只是针对Iterable进行了优化,因此需要创建一个迭代器。这是反编译后的Java代码表示形式:

  1. Iterable $receiver$iv = (Iterable)(new IntRange(1, 10));
  2. Iterator var1 = $receiver$iv.iterator();
  3. while(var1.hasNext()) {
  4. int element$iv = ((IntIterator)var1).nextInt();
  5. System.out.println(element$iv);
  6. }

该代码甚至比以前的示例效率更低,因为除了创建IntRange对象外, 你还必须还有创建一个IntIterator的开销。至少,这个会生成原生类型的值。

要对范围进行迭代,最好使用简单的for循环,而不是在其上调用forEach()函数,以避免迭代器对象的开销。

迭代: collection indices

Kotlin标准库提供了内置索引扩展属性,以生成数组索引和Collection索引的区间。

  1. val list = listOf("A", "B", "C")
  2. for (i in list.indices) {
  3. println(list[i])
  4. }

令人惊讶的是,遍历 indices 的代码也被编译为优化的代码

  1. List list = CollectionsKt.listOf(new String[]{
  2. "A", "B", "C"});
  3. int i = 0;
  4. for(int var2 = ((Collection)list).size(); i < var2; ++i) {
  5. Object var3 = list.get(i);
  6. System.out.println(var3);
  7. }

在这里,我们可以看到根本没有创建IntRange对象,并且列表迭代尽可能高效。

这对于实现Collection的数组和类非常有效, 因此你可能会在自己定义类中定义自己的indices扩展,同时期望能达到相同的迭代性能.

  1. inline val SparseArray<*>.indices: IntRange
  2. get() = 0 until size()
  3. fun printValues(map: SparseArray<String>) {
  4. for (i in map.indices) {
  5. println(map.valueAt(i))
  6. }
  7. }

但是,在编译之后,我们可以看到效率不高,因为编译器无法智能地避免创建区间对象:

  1. public static final void printValues(@NotNull SparseArray map) {
  2. Intrinsics.checkParameterIsNotNull(map, "map");
  3. IntRange var10000 = RangesKt.until(0, map.size());
  4. int i = var10000.getFirst();
  5. int var2 = var10000.getLast();
  6. if(i <= var2) {
  7. while(true) {
  8. Object $receiver$iv = map.valueAt(i);
  9. System.out.println($receiver$iv);
  10. if(i == var2) {
  11. break;
  12. }
  13. ++i;
  14. }
  15. }
  16. }

相反,我建议直接在for循环中使用until()函数

  1. fun printValues(map: SparseArray<String>) {
  2. for (i in 0 until map.size()) {
  3. println(map.valueAt(i))
  4. }
  5. }

当遍历未实现Collection接口的自定义集合时,最好直接在for循环中编写自己的索引范围,而不是依靠函数或属性来生成区间,以避免分配区间对象。

我希望这些对你的阅读和对我的写作一样有趣。你可能会在以后看到更多相关内容,但是前三部分涵盖了我计划最初编写的所有内容。如果你喜欢,请分享给他人,谢谢!

总结

到这里,有关探索Kotlin性能开销的系列文章终于暂时告于完结,说下自己切身感受,翻译这个系列对我平时在用Kotlin开发时有了很大的帮助,可以写出更加高效优秀的代码。所以我觉得有必要把它翻译出来和大家共享。下一站,我们将进入Kotlin协程~~~

[译]探索Kotlin中隐藏的性能开销-Part 3

翻译说明:

原标题# Exploring Kotlin’s hidden costs — Part 3

原文地址: https://medium.com/@BladeCoder/exploring-kotlins-hidden-costs-part-2-324a4a50b70

原文作者: Christophe Beyls

代理属性和Range

在发布有关Kotlin编程语言的性能开销系列的前两篇文章之后,我收到了很多不错的反馈,甚至还包括 Jake Wharton 大神他自己。所以你还没看前两篇文章,千万不要错过哦。

在第3部分中,我们将揭开更多有关Kotlin编译器的秘密,并提供如何编写更高效代码的新技巧。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-3EaYp3g2-1572187598551)(https://miro.medium.com/max/646/1\*-iKupZ7diZBEzTw87Bkaxg.jpeg)\]

一、代理属性

代理属性是一种其getter和可选的setter的内部实现可由代理的外部对象提供的属性。它可以允许复用自定义属性的内部实现。

  1. class Example {
  2. var p: String by Delegate()
  3. }

这个代理对象必须实现一个 operator getVlue()函数,以及一个 setValue()函数来用于属性的读/写. 这些函数将接收包含对象实例 以及属性的metadata元数据 作为额外参数(比如它的属性名)。

当类中声明一个代理属性时,编译将生成以下代码(下面是反编译后的Java代码):

  1. public final class Example {
  2. @NotNull
  3. private final Delegate p$delegate = new Delegate();
  4. // $FF: synthetic field
  5. static final KProperty[] $$delegatedProperties = new KProperty[]{
  6. (KProperty)Reflection.mutableProperty1(new MutablePropertyReference1Impl(Reflection.getOrCreateKotlinClass(Example.class), "p", "getP()Ljava/lang/String;"))};
  7. @NotNull
  8. public final String getP() {
  9. return this.p$delegate.getValue(this, $$delegatedProperties[0]);
  10. }
  11. public final void setP(@NotNull String var1) {
  12. Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
  13. this.p$delegate.setValue(this, $$delegatedProperties[0], var1);
  14. }
  15. }

一些静态属性metadata元数据被添加到类中。代理将在类的构造器中进行初始化,然后在每次读取或写入属性时都调用该代理。

代理实例

在上述例子中,将会创建一个新的代理对象的实例来实现该属性。当代理实例是有状态的时候, 这就是必需的,例如在计算本地缓存属性的值时.

  1. class StringDelegate {
  2. private var cache: String? = null
  3. operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
  4. var result = cache
  5. if (result == null) {
  6. result = someOperation()
  7. cache = result
  8. }
  9. return result
  10. }
  11. }

如果还需要通过其构造函数传递的额外参数,则还需要创建一个新的代理实例:

  1. class Example {
  2. private val nameView by BindViewDelegate<TextView>(R.id.name)
  3. }

但是在某些情况下,只需要一个代理实例就可以实现任意属性: 当代理实例是无状态的时候,并且它执行所需的唯一变量就是对象实例和属性名称(然而这些编译器都直接提供了)。在这种情况下,可以通过将代理实例声明成object对象表达式而不是一个来使得成为单例

例如,下面的代理单例实例检索其标记名称与Android Activity 中的属性名称来匹配Fragment.

  1. object FragmentDelegate {
  2. operator fun getValue(thisRef: Activity, property: KProperty<*>): Fragment? {
  3. return thisRef.fragmentManager.findFragmentByTag(property.name)
  4. }
  5. }

同样,任意的对象都可以扩展成代理。此外getValue()setValue()还可以声明成扩展函数。Kotlin中已经提供了内置的扩展函数,例如允许将MapMutableMap实例作为代理实例,并将属性的名称作为key.

如果你选择在同一个类中实现多个属性复用同一个局部代理实例的话,那么需要在类的构造器中初始化此实例。

注意: 从Kotlin1.1开始,也可以在函数中声明局部变量作为代理属性。那么在这种情况下,代理实例可以延迟初始化,直到在函数中声明变量为止。

在类中声明的每个代理属性都涉及到其关联的代理对象创建的性能开销,并向该类中添加一些metadata元数据。必要的时候,可以尝试为不同属性复用同一个代理实例。在你声明大量代理属性的时候,还需要考虑代理属性是否你的最佳选择。

泛型代理

还可以以泛型的方式声明代理函数,因此同一个代理类可以用任意的属性类型。

  1. private var maxDelay: Long by SharedPreferencesDelegate<Long>()

但是,如果像上面例子那样使用具有原生类型属性的泛型代理的话,即便声明的原生类型为非null,每次读取或写入该属性时都避免不了装箱和拆箱的发生

对于非null原生类型的代理属性,最好使用为该特定值类型创建特定的代理类,而不是泛型代理,以避免在每次访问该属性时产生的装箱开销

标准库代理: lazy()

Kotlin内置了一些标准库代理函数来覆盖常见的情况,例如 Delegates.notNull(),Delegates.observable()lazy().

lazy(initializer: () -> T) 是一个为只读属性返回代理对象的函数,该属性是通过在其首次被读取的时,lazy函数参数lambda initializer执行来初始化的。

  1. private val dateFormat: DateFormat by lazy {
  2. SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
  3. }

这是一种将昂贵的初始化操作延迟到实际需要使用之前的巧妙方法,可以在保持代码可读性的同时又提高了性能。

需要注意到的是,lazy()函数不是内联函数,并且作为参数传递的lambda将编译成独立的Function类,并且不会在返回的代理对象内进行内联。

通常会被人忽略的是lazy()另一重载函数实际上还隐藏一个可选的模式参数来确定应该返回3种不同类型的代理中的一种:

  1. public fun <T> lazy(initializer: () -> T): Lazy<T> = SynchronizedLazyImpl(initializer)
  2. public fun <T> lazy(mode: LazyThreadSafetyMode, initializer: () -> T): Lazy<T> =
  3. when (mode) {
  4. LazyThreadSafetyMode.SYNCHRONIZED -> SynchronizedLazyImpl(initializer)
  5. LazyThreadSafetyMode.PUBLICATION -> SafePublicationLazyImpl(initializer)
  6. LazyThreadSafetyMode.NONE -> UnsafeLazyImpl(initializer)
  7. }

默认的模式是 LazyThreadSafetyMode.SYNCHRONIZED 将执行相对开销昂贵的双重锁的检查,这是为了保证在多线程环境下读取属性时,初始化块可以安全运行。

如果你明确知道当前环境是单线程(例如主线程)访问属性,那么可以通过显式使用 LazyThreadSafetyMode.NONE 来完全避免双重锁的检查所带来昂贵的开销。

  1. val dateFormat: DateFormat by lazy(LazyThreadSafetyMode.NONE) {
  2. SimpleDateFormat("dd-MM-yyyy", Locale.getDefault())
  3. }

使用lazy()代理可以按需延迟昂贵的初始化,此外可以指定线程安全的模式以避免不必要的双重锁检查。

二、Ranges(区间)

区间是一种用于表示Kotlin中的一组有限值的特殊表达式。这些值可以是任意Comparable类型。这些表达式由创建用于实现ClosedRange对象的函数形成。用于创建区间的主要函数是 ..操作符。

区间包含的测试

区间表达式主要目的是使用in!in 运算符来判断是否包含某个值

  1. if (i in 1..10) {
  2. println(i)
  3. }

该实现特地针对非null原生类型区间(有: Int, Long, Byte, Short, Float, Double或Char)进行了优化,因此上面例子可以高效编译成如下形式:

  1. if(1 <= i && i <= 10) {
  2. System.out.println(i);
  3. }

性能开销几乎为0,没有额外的对象分配。区间也可以和任意其他非原生Comparable类型一起使用。

  1. if (name in "Alfred".."Alicia") {
  2. println(name)
  3. }

在Kotlin 1.1.50之前,编译以上示例时始终会创建一个临时的ClosedRange对象。但是从1.1.50之后,已经对它的实现进行了优化,以避免Comparable类型额外开销分配:

  1. if(name.compareTo("Alfred") >= 0) {
  2. if(name.compareTo("Alicia") <= 0) {
  3. System.out.println(name);
  4. }
  5. }

此外,区间检查还包括应用再 when 表达式中

  1. val message = when (statusCode) {
  2. in 200..299 -> "OK"
  3. in 300..399 -> "Find it somewhere else"
  4. else -> "Oops"
  5. }

这使代码比一系列if {...} else if {...}语句更具可读性,并且效率更高。

但是,在区间包含检查中,当区间的声明之间至少存在一个间接过程时,会有一个小的性能开销。 比如下面这段Kotlin代码:

  1. private val myRange get() = 1..10
  2. fun rangeTest(i: Int) {
  3. if (i in myRange) {
  4. println(i)
  5. }
  6. }

上述代码会造成在编译后额外创建一个IntRange对象:

  1. private final IntRange getMyRange() {
  2. return new IntRange(1, 10);
  3. }
  4. public final void rangeTest(int i) {
  5. if(this.getMyRange().contains(i)) {
  6. System.out.println(i);
  7. }
  8. }

即使将属性getter声明成内联函数也不能避免创建IntRange对象。在这种情况下,Kotlin 1.1编译器已经改进了。 由于这些特定的区间类存在,至少在比较原生类型时不会出现装箱过程。

尝试在没有间接声明过程区间检查中使用直接声明区间的方式,来避免额外区间对象的创建分配,另外,可以将它们声明成常量以此来复用他们。

迭代: for循环

整数类型区间(除Float或Double之外的任何原生类型的区间)也是级数: 可以对其进行迭代。这允许用较短的语法替换经典的Java for循环。

  1. for (i in 1..10) {
  2. println(i)
  3. }

这可以以零开销方式编译为可比较的优化代码:

  1. int i = 1;
  2. for(byte var2 = 11; i < var2; ++i) {
  3. System.out.println(i);
  4. }

如果向后迭代,请使用 downTo() 中缀函数来替代

  1. for (i in 10 downTo 1) {
  2. println(i)
  3. }

同样,使用此构造进行编译后的开销为零:

  1. int i = 10;
  2. byte var1 = 1;
  3. while(true) {
  4. System.out.println(i);
  5. if(i == var1) {
  6. return;
  7. }
  8. --i;
  9. }

还有一个有用的until()中缀函数可以迭代直到但不包括区间上限值。

  1. for (i in 0 until size) {
  2. println(i)
  3. }

当本文的原始版本发布时,调用此函数用于生成次优代码。自Kotlin 1.1.4起,情况已大大改善,并且编译器现在生成等效的Java for循环:

  1. int i = 0;
  2. for(int var2 = size; i < var2; ++i) {
  3. System.out.println(i);
  4. }

但是,其他迭代变体的优化效果也不佳

这是另一种使用reversed() 函数与区间组合的方法,可以向后迭代并产生与downTo()完全相同的结果。

  1. for (i in (1..10).reversed()) {
  2. println(i)
  3. }

不幸的是,生成的编译代码就不那么漂亮:

  1. IntProgression var10000 = RangesKt.reversed((IntProgression)(new IntRange(1, 10)));
  2. int i = var10000.getFirst();
  3. int var3 = var10000.getLast();
  4. int var4 = var10000.getStep();
  5. if(var4 > 0) {
  6. if(i > var3) {
  7. return;
  8. }
  9. } else if(i < var3) {
  10. return;
  11. }
  12. while(true) {
  13. System.out.println(i);
  14. if(i == var3) {
  15. return;
  16. }
  17. i += var4;
  18. }

将会创建一个临时的IntRange对象来表示区间,然后再创建另一个IntProgression对象来反转第一个对象的值。

事实上,创建一个progression的以上功能任何组合都会生成类似的代码,涉及到创建至少两个轻量级progression对象的小开销。

此规则也适用于使用step()中缀函数来修改progression, 即使步长是1:

  1. for (i in 1..10 step 2) {
  2. println(i)
  3. }

附带说明下,当生成的代码读取IntProgression的最后一个属性时,这将执行少量计算,以通过考虑边界和步长来确定区间的确切最后一个值。在上面的示例中,最后一个值应该为9。

若要在for循环中进行迭代,最好使用区间表达式,该区间表达式只涉及到对 ..downTo()untill()的单个函数调用,以避免创建临时progression对象的开销。

迭代: for-each()

与其使用for循环,不如尝试在区间上使用forEach()内联扩展函数来达到相同的结果。

  1. (1..10).forEach {
  2. println(it)
  3. }

但是,如果您仔细查看此处使用的forEach()函数的签名,你会注意到,它并没有针对区间进行优化,而只是针对Iterable进行了优化,因此需要创建一个迭代器。这是反编译后的Java代码表示形式:

  1. Iterable $receiver$iv = (Iterable)(new IntRange(1, 10));
  2. Iterator var1 = $receiver$iv.iterator();
  3. while(var1.hasNext()) {
  4. int element$iv = ((IntIterator)var1).nextInt();
  5. System.out.println(element$iv);
  6. }

该代码甚至比以前的示例效率更低,因为除了创建IntRange对象外, 你还必须还有创建一个IntIterator的开销。至少,这个会生成原生类型的值。

要对范围进行迭代,最好使用简单的for循环,而不是在其上调用forEach()函数,以避免迭代器对象的开销。

迭代: collection indices

Kotlin标准库提供了内置索引扩展属性,以生成数组索引和Collection索引的区间。

  1. val list = listOf("A", "B", "C")
  2. for (i in list.indices) {
  3. println(list[i])
  4. }

令人惊讶的是,遍历 indices 的代码也被编译为优化的代码

  1. List list = CollectionsKt.listOf(new String[]{
  2. "A", "B", "C"});
  3. int i = 0;
  4. for(int var2 = ((Collection)list).size(); i < var2; ++i) {
  5. Object var3 = list.get(i);
  6. System.out.println(var3);
  7. }

在这里,我们可以看到根本没有创建IntRange对象,并且列表迭代尽可能高效。

这对于实现Collection的数组和类非常有效, 因此你可能会在自己定义类中定义自己的indices扩展,同时期望能达到相同的迭代性能.

  1. inline val SparseArray<*>.indices: IntRange
  2. get() = 0 until size()
  3. fun printValues(map: SparseArray<String>) {
  4. for (i in map.indices) {
  5. println(map.valueAt(i))
  6. }
  7. }

但是,在编译之后,我们可以看到效率不高,因为编译器无法智能地避免创建区间对象:

  1. public static final void printValues(@NotNull SparseArray map) {
  2. Intrinsics.checkParameterIsNotNull(map, "map");
  3. IntRange var10000 = RangesKt.until(0, map.size());
  4. int i = var10000.getFirst();
  5. int var2 = var10000.getLast();
  6. if(i <= var2) {
  7. while(true) {
  8. Object $receiver$iv = map.valueAt(i);
  9. System.out.println($receiver$iv);
  10. if(i == var2) {
  11. break;
  12. }
  13. ++i;
  14. }
  15. }
  16. }

相反,我建议直接在for循环中使用until()函数

  1. fun printValues(map: SparseArray<String>) {
  2. for (i in 0 until map.size()) {
  3. println(map.valueAt(i))
  4. }
  5. }

当遍历未实现Collection接口的自定义集合时,最好直接在for循环中编写自己的索引范围,而不是依靠函数或属性来生成区间,以避免分配区间对象。

我希望这些对你的阅读和对我的写作一样有趣。你可能会在以后看到更多相关内容,但是前三部分涵盖了我计划最初编写的所有内容。如果你喜欢,请分享给他人,谢谢!

总结

到这里,有关探索Kotlin性能开销的系列文章终于暂时告于完结,说下自己切身感受,翻译这个系列对我平时在用Kotlin开发时有了很大的帮助,可以写出更加高效优秀的代码。所以我觉得有必要把它翻译出来和大家共享。下一站,我们将进入Kotlin协程~~~

1635c3fb0ba21ec1_w_430_h_430_f_jpeg_s_39536

欢迎关注Kotlin开发者联盟,这里有最新Kotlin技术文章,每周会不定期翻译一篇Kotlin国外技术文章。如果你也喜欢Kotlin,欢迎加入我们~~~

Kotlin系列文章,欢迎查看:

Kotlin邂逅设计模式系列:

  • 当Kotlin完美邂逅设计模式之单例模式(一)

数据结构与算法系列:

  • 每周一算法之二分查找(Kotlin描述)
  • 每周一数据结构之链表(Kotlin描述)

翻译系列:

  • [译] [译]探索Kotlin中隐藏的性能开销-Part 2
  • [译] [译]探索Kotlin中隐藏的性能开销-Part 1
  • [译] Kotlin中关于Companion Object的那些事
  • [译]记一次Kotlin官方文档翻译的PR(内联类)
  • [译]Kotlin中内联类的自动装箱和高性能探索(二)
  • [译]Kotlin中内联类(inline class)完全解析(一)
  • [译]Kotlin的独门秘籍Reified实化类型参数(上篇)
  • [译]Kotlin泛型中何时该用类型形参约束?
  • [译] 一个简单方式教你记住Kotlin的形参和实参
  • [译]Kotlin中是应该定义函数还是定义属性?
  • [译]如何在你的Kotlin代码中移除所有的!!(非空断言)
  • [译]掌握Kotlin中的标准库函数: run、with、let、also和apply
  • [译]有关Kotlin类型别名(typealias)你需要知道的一切
  • [译]Kotlin中是应该使用序列(Sequences)还是集合(Lists)?
  • [译]Kotlin中的龟(List)兔(Sequence)赛跑

原创系列:

  • 教你如何完全解析Kotlin中的注解
  • 教你如何完全解析Kotlin中的类型系统
  • 如何让你的回调更具Kotlin风味
  • Jetbrains开发者日见闻(三)之Kotlin1.3新特性(inline class篇)
  • JetBrains开发者日见闻(二)之Kotlin1.3的新特性(Contract契约与协程篇)
  • JetBrains开发者日见闻(一)之Kotlin/Native 尝鲜篇
  • 教你如何攻克Kotlin中泛型型变的难点(实践篇)
  • 教你如何攻克Kotlin中泛型型变的难点(下篇)
  • 教你如何攻克Kotlin中泛型型变的难点(上篇)
  • Kotlin的独门秘籍Reified实化类型参数(下篇)
  • 有关Kotlin属性代理你需要知道的一切
  • 浅谈Kotlin中的Sequences源码解析
  • 浅谈Kotlin中集合和函数式API完全解析-上篇
  • 浅谈Kotlin语法篇之lambda编译成字节码过程完全解析
  • 浅谈Kotlin语法篇之Lambda表达式完全解析
  • 浅谈Kotlin语法篇之扩展函数
  • 浅谈Kotlin语法篇之顶层函数、中缀调用、解构声明
  • 浅谈Kotlin语法篇之如何让函数更好地调用
  • 浅谈Kotlin语法篇之变量和常量
  • 浅谈Kotlin语法篇之基础语法

Effective Kotlin翻译系列

  • [译]Effective Kotlin系列之考虑使用原始类型的数组优化性能(五)
  • [译]Effective Kotlin系列之使用Sequence来优化集合的操作(四)
  • [译]Effective Kotlin系列之探索高阶函数中inline修饰符(三)
  • [译]Effective Kotlin系列之遇到多个构造器参数要考虑使用构建器(二)
  • [译]Effective Kotlin系列之考虑使用静态工厂方法替代构造器(一)

实战系列:

  • 用Kotlin撸一个图片压缩插件ImageSlimming-导学篇(一)
  • 用Kotlin撸一个图片压缩插件-插件基础篇(二)
  • 用Kotlin撸一个图片压缩插件-实战篇(三)
  • 浅谈Kotlin实战篇之自定义View图片圆角简单应用

发表评论

表情:
评论列表 (有 0 条评论,116人围观)

还没有评论,来说两句吧...

相关阅读