一篇文章弄懂kotlin的扩展方法
作者:DROID 发布时间:2022-06-05 08:06:20
Usage
扩展函数是 kotlin 的又一杀手锏功能,能够在不修改源码的基础上,扩展某些类的能力,方便开发。
例如这里演示了给 String 添加一个获取第一个元素的方法。
fun String.first(): Char {
if (isEmpty()) {
throw NoSuchElementException("String is empty")
}
return this[0]
}
fun main(args: Array<String>) {
println("Hello,World".first())
}
这里需要额外注意的地方在于扩展函数的方法体中,是能够直接访问扩展对象 public 的变量的。例如上面的方法里面,我们也可以这么写:
fun String.first(): Char {
if (length < 1) {
throw NoSuchElementException("String is empty")
}
return this[0]
}
通过 this 可以在方法内,访问扩展对象,这里就是通过 this[0] 拿到第一个字符的。
Under in hood
看上去很厉害哈,但他的原理却非常简单。我们要时刻记住,kotlin JVM 是基于 JVM 开发的,kotlin 源码最后会变成字节码而后被运行。当遇到语法上不太懂的地方,直接反编译字节码,或者 Decompile 成 Java 方法,就能洞察里面的玄机。
我们将上述代码,Decompile 成 Java 后,就能发现里面的秘密。
public static final char first(@NotNull String $this$first){
Intrinsics.checkParameterIsNotNull($this$first, "$this$first");
if ($this$first.length() < 1) {
throw (Throwable)(new NoSuchElementException("String is empty"));
} else {
return $this$first.charAt(0);
}
}
原来是生成了一个 public static final 的方法呀,不过这个生成是 kotlin 提供的语法糖,帮我们完成的。看到这个代码,也解释了为什么在扩展对象方法内部,能够访问到扩展对象的 public 成员。
重载与多态
扩展方法能否被继承呢,或者重载呢?我们来看看例子
open class Animal
class Dog : Animal()
fun Animal.desc() = "Animal"
fun Dog.desc() = "Dog"
fun main(args: Array<String>) {
println(Dog().desc())
var animal: Animal = Dog()
println(animal.desc())
}
// output:
// Dog
// Animal
如果扩展方法能够被重载,那么两次都应该输出 Dog,我们还是和前面方法一样,来看看真相。
@NotNull
public static final String desc(@NotNull Animal $this$desc) {
Intrinsics.checkParameterIsNotNull($this$desc, "$this$desc");
return "Animal";
}
@NotNull
public static final String desc(@NotNull Dog $this$desc) {
Intrinsics.checkParameterIsNotNull($this$desc, "$this$desc");
return "Dog";
}
可以看到实际生成了两个 desc 方法,里面的参数不动,所以这个方法的调用,只与扩展对象本身有关系,在编译时已经确定,不存在多态。
扩展属性
这是一个很神奇的设定,kotlin 并不能真的给扩展对象添加一个属性,而只是提供了一个语法糖,什么意思呢?我们具体看看下面这个例子。
var String.first: Char
get() {
if (isEmpty()) {
throw NoSuchElementException(“String is empty”)
}
return this[0]
}
set(value) {
println(“set value to $value”)
}
fun main() {
“Hello, World”.first = ‘G'
println(“Hello,World”.first)
}
我们扩展了 kotlin 的属性,添加了一个 first。我们可以分别给这个所谓的 first 属性,注意是所谓的,添加 get 和 set 方法。然后我们可以通过 = 和 . 来调用 set 和 get 方法,就像 main 方法中那样。但实际上,最后并没有生成 first 属性,我们来看看反编译过后的代码。
public static final Char getFirst(@NotNull String $this$first) {
Intrinsics.checkParameterIsNotNull($this$first, "$this$first");
CharSequence var1 = (CharSequence)$this$first;
boolean var2 = false;
if (var1.length() == 0) {
throw (Throwable)(new NoSuchElementException("String is empty"));
} else {
return $this$first.charAt(0);
}
}
public static final void setFirst(@NotNull String $this$first, char value) {
Intrinsics.checkParameterIsNotNull($this$first, "$this$first");
String var2 = "set value to " + value;
boolean var3 = false;
System.out.println(var2);
}
看到没有,实际上只是添加了 setFirst 和 getFirst 两个方法,并没有实际的属性添加上去。这也是 kotlin 提供给我们的语法糖之一,糖要吃,但也要小心蛀牙哦!
好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。
来源:https://woaitqs.cc/2019/09/08/kotlin-extension-method/
猜你喜欢
- 后台服务端import java.io.IOException;import java.io.InputStream;import java
- 本文实例主要进行java Timer(定时调用、固定时间执行)测试,具体实现代码如下。测试1当任务执行时间小于重复执行的间隔时间代码:pub
- 基础环境SpringBoot、Maven代码实现1.添加依赖<!--二维码生成 --><dependency&
- 本文主要关注如何使用mybatis/mybatis plus连接SQL Server数据库,因此将省略其他项目配置、代码。框架选择应用框架:
- 本文实例为大家分享了基于C#实现网页爬虫的详细代码,供大家参考,具体内容如下HTTP请求工具类:功能:1、获取网页html2、下载网络图片u
- SpringBoot配置文件的替换使用spring.profiles.active在工作中,测试或上线的时候一定会遇到的问题就是修改配置。一
- 一.IDEA开发环境1.pom文件设置<properties> <maven.compiler.
- 声明:下面的实例全部在linux下尝试,window下未尝试。有兴趣者可以试一下。文章针c初学者。c语言的强符号和弱符号是c初学者经常容易犯
- 这两天因为要做一个随机的地图生成系统,所以一直在研究随机迷宫生成算法,好吧,算是有一点小小的成果。随机迷宫生成我自己的理解简而言之分为以下几
- 我们还是用一个小例子来看看自定义View和自定义属性的使用,带大家来自己定义一个带进度的圆形进度条,我们还是先看一下效果吧从上面可以看出,我
- 算法描述堆排序算法的描述如下:将待排序的数组调整为最大堆,此时未排序的长度 N 为数组的长度,调整的过程就是倒序将数组的
- using System;using System.Collections.Generic;using System.Web.Script.
- 传输层安全性协议(英语:Transport Layer Security,缩写作 TLS),及其前身安全套接层(Secure Sockets
- 一、直接插入排序基本思想:将一个记录插入到已排序的有序表中,使插入后的表仍然有序对初始关键字{49 38 65 97 76 13 27 49
- using System;using System.Data;using System.Data.OleDb;namespace ZFSof
- 本文设计一个简单的班级管理系统,满足如下要求:1、设计学生类Student,包含学号(String型)、姓名(String型)、
- BeanFactory接口:IoC容器的顶级接口,是IoC容器的最基础实现,也是访问Spring容器的根接口,负责对bean的创建,访问等工
- C#中using指令的用法1.using指令using + 命名空间名字,这样可以在程序中直接用命令空间中的类型,而不必指定类型的详细命名空
- 什么是WebSocket WebSocket是一种在单个TCP连接上进行全双工通信的协议。WebSocket使得客户端和服务器之间
- 详解Kotlin的空指针处理Kotlin的空指针处理相比于java有着极大的提高,可以说是不用担心出现NullPointerExceptio