kotlin映射c语言指针,Kotlin难点解析:extension和this指针

扩展(extension)是Kotlin语言中使用非常简单的一个特性。这篇文章并不是要讲解扩展的基本用法,而是解决在一些复杂场景中,扩展容易让人产生迷惑的一些问题。除了扩展,本篇文章还将讲解this指针在Kotlin语言中的基础用法。

1460000014376500?w=480&h=360

扩展函数难点解析

大多数场景下,你都能轻松搞定Kotlin扩展。可是,看看下面这个题目,你还能脱口而出,告诉我答案是什么吗?

open class E {

}

open class E1: E() {

}

open class A {

open fun E.f() {

println("E.f in A")

}

open fun E1.f() {

println("E1.f in A")

}

fun call(e: E) {

e.f()

}

}

class A1: A() {

override fun E.f() {

println("E.f in A1")

}

override fun E1.f() {

println("E1.f in A1")

}

}

fun main(args: Array) {

// a)

A().call(E())

// b)

A1().call(E())

// c)

A().call(E())

// d)

A().call(E1())

}

问题:请告诉a,b,c,d位置代码执行的输出结果是什么?

对于这个问题,恐怕你在纸上写写画画半天也不一定能给出正确答案吧。关于这个问题,其实我之前的一篇文章 [ [Kotlin] Lambda and Extension](https://www.jianshu.com/p/d7a... 中有提到过。可是,我认为这篇文章关于这部分的解释不够清晰,有必要再详细阐述一次。

Ok,let's started。

为了解决这个问题,官方提出了两个新的概念:dispatch receiver和extension receiver。

dispatch receiver:中文翻译为分发接收者。所谓的分发接收者,就是声明这个扩展方法所在的类。即:在哪个类中声明,那个类就是你的分发接收者。

extension receiver:中文翻译为扩展接收者。所谓的扩展接收者,就是你实际扩展的那个类。举个例子:你针对Int类扩展了一个方法add,这个add方法的扩展接收者就是Int类实例。

为了简化,这里我们将dispatch receiver简称为DR,将extension receiver简称为ER。

还记得多态的概念吗?多态是一种运行时概念,即对象的类型要等到运行时才能最终确定。因此,一些语言中也将多态叫做类型延迟加载。解决上面这个问题我们需要关注就是扩展函数是否会产生多态行为。

这里我们将产生多态行为的技术叫做动态解析,与之相反的行为称之为静态解析。

为了解决上面的问题,你需要记住下面这个规则:

DR类型是动态解析的

与之相反,ER类型是静态解析的

先看上面例子的a、b部分,很显然:

a代码中f函数的DR是类A,ER是类E

b代码中f函数的DR是类A1,ER是类E

参照上面的规则,由于DR类型是动态解析的。在A1类中我们重写了E的扩展函数f,运行时最终会执行A1类中扩展的f方法。a部分很明显会输出A类中扩展的f方法。因此,最终的输出结果如下:

E.f in A

E.f in A1

继续看c、d部分,c、d部分的DR都是A,而对于ER,c、d分别是E、E1。参照上面的规则,ER是静态解析的。在call方法声明的地方,我们传入的对象类型是E,这就决定了无论扩展方法是来自E还是其子类,将始终执行E类的扩展方法。因此,c、d部分将输出同样的结果:

E.f in A

E.f in A

由此可见,如果你牢记上述两条规则,解决问题将变得非常容易。为了加强你的记忆,我用一个表格总结上面的知识点:

-

DR

ER

概念

扩展方法声明所在的类

声明扩展方法的类

解析方式

动态解析

静态解析

PS:由于新版本Kotlin中针对扩展函数也加入了override关键字,这非常有助于DR和ER的理解。如果你在使用Kotlin,强烈建议你更新到最新版本。

1460000014376501?w=1240&h=1240

不太一样的this指针

在Java语言中,如果你在内部类中需要外部类的引用可以将this写在类名后面。可是,试试看Kotlin,果断不行。

为了获得外部类的引用,Kotlin语言引入了@符号。举个例子:

class Outer {

inner class Inner {

fun f() {

println(this@Outer)

}

}

}

可以看到,为了获取外部类的引用,只需要在@后面接外部类的名称即可。

如果对应一个扩展函数,this引用指向是什么呢?先说答案,扩展函数中的this指针指向ER,即实际扩展的那个类对象。

fun Outer.foo() {

println(this)

}

这里的this指向foo函数的接收者Outer类实例。

this指针还有一种场景是用在lambda表达式中,这是一种比较特殊的使用场景。lambda表达式本身没有任何接收者,如果是在全局声明一个lambda表达式,将不能使用this指针。而如果是在某个类或者扩展方法中使用this指针,将指向实际所在类或者扩展方法的接收者。

如果你习惯了Kotlin语言的这种表达方式,this指针的指向就不再是一个问题了。在你习惯这种用法之前,我用一个表格简单总结一下this指针的用法:

位置

指向

类中

默认指向当前类实例,使用@操作符指向具体外部类实例

扩展函数

默认指向扩展函数的接收者

lambda表达式

默认指向实际所在类实例或所在扩展函数的接收者

总结

关于扩展,大多数情况下,你不会遇到文章开头那种复杂的情况。如果遇到了这种情况,只要清楚地区分DR和ER,并牢记DR和ER的解析方式,就能轻松应对了。对于this指针,与Java语言不一样的地方是,为了引用具体类的实例,Kotlin语言使用@符号。个人认为,这种表述方式更自然。如果遇到某些比较复杂的情况,只需要弄清楚接收者,问题就引刃而解了。

欢迎加入Kotlin交流群

如果你也喜欢Kotlin语言,欢迎加入我的Kotlin交流群: 329673958 ,一起来参与Kotlin语言的推广工作。

编程,我们是认真的!

关注欧阳锋工作室公众号,你想要的都在这里:

1460000013572045?w=430&h=430

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值