Kotlin Extension — Method

简单来说,Extension就是Kotlin版的 Decorator(装饰者模式)

【Example】当我们使用Extension特性的时候,需要指定一个Receiver, 例如

fun C.foo(){
    ...
}

Receiver就是C。

1. Extensions resolving

假设我们写一个简单的类Parent (in Parent.kt):

class Parent {
    val value : Int = 1
}

为了方便比较,再一些简单的类Other,同时在Other中写一个Parent的Extension markValue() (in Other.kt):

class Other {
    val const = 2

    fun Parent.markValueOne() : Int {
        return this.value + 1
    }
}

fun Parent.markValueTwo() : Int {
    return this.value + 2
}

在经过kotlinc编译之后,通过javap查看相关的class文件,markValueOnemarkValueTwo 都被编译成了final的方法。
Other.class

public final class com.maxtropy.viewtest.Other {
  public final int getConst();
    Code:
       0: aload_0
       1: getfield      #11                 // Field const:I
       4: ireturn

  public final int markValueOne(com.maxtropy.viewtest.Parent);
    Code:
       0: aload_1
       1: ldc           #18                 // String $receiver
       3: invokestatic  #24                 // Method kotlin/jvm/internal/Intrinsics.checkParameterIsNotNull:(Ljava/lang/Object;Ljava/lang/String;)V
       6: aload_1
       7: invokevirtual #29                 // Method com/maxtropy/viewtest/Parent.getValue:()I
      10: iconst_1
      11: iadd
      12: ireturn

  public com.maxtropy.viewtest.Other();
    Code:
       0: aload_0
       1: invokespecial #34                 // Method java/lang/Object."<init>":()V
       4: aload_0
       5: iconst_2
       6: putfield      #11                 // Field const:I
       9: return
}

OtherKt.class

public final class com.maxtropy.viewtest.OtherKt {
  public static final int markValueTwo(com.maxtropy.viewtest.Parent);
    Code:
       0: aload_0
       1: ldc           #9                  // String $receiver
       3: invokestatic  #15                 // Method kotlin/jvm/internal/Intrinsics.checkParameterIsNotNull:(Ljava/lang/Object;Ljava/lang/String;)V
       6: aload_0
       7: invokevirtual #21                 // Method com/maxtropy/viewtest/Parent.getValue:()I
      10: iconst_2
      11: iadd
      12: ireturn
}

NOTE: 需要注意的是虽然markValueOne的写法能够通过编译,但如果写的.kt,是拿不到markValueOne方法引用的。(令人奇怪的是写.java却能够拿到markValueOne方法的引用 ???) 很明显的可以看出来,无论是哪种写法,都会将Parent类型实例作为函数的第一个入参传进方法当中。而在方法中用到的this指针,指的便是这一个Parent实例,而不是传统Java编程习惯中this永远指向的是当前被调用实例方法对应的实例。

2. 实例方法与Extension方法同名

倘若在Receiver类型中和Extension中声明了同名的方法呢?,例如:

在Parent中声明了一个方法markValueTwo()

class Parent {
    val value : Int = 1

    fun markValueTwo(): Int {
        return this.value + 99
    }
}

然后在另一个类Other的.kt文件中使用Extension并让Extension的方法同Parent中的方法名一样

class Other {
    fun getParentMarkValue(): Int {
        return Parent().markValueTwo()
    }
}

fun Parent.markValueTwo(): Int {
    return this.value + 2
}

经过kotlinc的编译(在编译过程中其实就会有warning), 会发现Other.class文件的中:

Compiled from "Other.kt"
public final class com.maxtropy.viewtest.Other {
  public final int getParentMarkValue();
    Code:
       0: new           #8                  // class com/maxtropy/viewtest/Parent
       3: dup
       4: invokespecial #12                 // Method com/maxtropy/viewtest/Parent."<init>":()V
       7: invokevirtual #15                 // Method com/maxtropy/viewtest/Parent.markValueTwo:()I
      10: ireturn

  public com.maxtropy.viewtest.Other();
    Code:
       0: aload_0
       1: invokespecial #18                 // Method java/lang/Object."<init>":()V
       4: return
}

在字节码执行Parent实例的markValueTwo()方法时调用的是虚方法,毫无疑问调用的其实就是实例中对应的实例方法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,969评论 19 139
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,776评论 0 9
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,767评论 18 399
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 770评论 0 2
  • 不重要的废话 前段时间看了一遍《Programming Kotlin》,主要目的是想提高自己的英文阅读能力,能力提...
    珞泽珈群阅读 3,395评论 1 7