扩展函数
Kotlin扩展函数能够对⼀个类扩展新功能⽽⽆需继承该类或者使⽤像装饰者这样的设计模式。利用他可以减少很多代码,提升我们的开发效率。
扩展函数的声明
声明⼀个函数,然后让被扩展的类型作为函数的前缀就可以了,比如下面对TextView的扩展
fun TextView.setDrawableLeft(drawableId: Int?) {
val drawable = ContextCompat.getDrawable(this.context, drawableId)
if (drawable == null) {
setCompoundDrawables(
null,
compoundDrawables[1],
compoundDrawables[2],
compoundDrawables[3]
)
} else {
drawable.apply {
setBounds(0, 0, minimumWidth, minimumHeight)
setCompoundDrawables(
drawable,
compoundDrawables[1],
compoundDrawables[2],
compoundDrawables[3]
)
}
}
}
然后我们就可以使用TextView的实例调用setDrawableLeft()方法。虽然看上去好像我们为TextView新增了一个方法,但是其实扩展是静态的,他并没有为扩展类型中插入新的方法
那么他是如何实现的呢,我们反编译以下class文件,查看对应的java文件就明白了
扩展函数原理
public static final void setDrawableLeft(@NotNull TextView $this$setDrawableLeft, @Nullable Integer drawableId)
{
//省略其他代码
}
原来是给我们生成了一个对应的java静态方法,第一个参数就是接受者类型的对象,所以在方法内部可以访问这个类中的成员。
既然是生成了java代码,那么这个方法就可以被其他java代码访问
TextViewKt.setDrawableLeft(textview,drawableId)
扩展属性
和扩展函数类似,这种扩展也是静态的,并没有给原来的类添加新的属性
扩展属性的声明
val StringBuilder.lastChar: String
get() {
if (this.isEmpty()) {
return ""
}
return this.substring(length - 1)
}
声明方式和扩展函数也类似,需要一个接受者类型,在作用域类,this就代表这个接受者对象,由于这里lastChar我们定义的是val,所以只用定义get()方法就可以了,如果是var类型,还需要定义set()方法,扩展字段的使用和正常字段一样。
val s = StringBuilder("value")
println(s.lastChar)
扩展属性原理
继续反编译查看对应的java源码
public static final String getLastChar(@NotNull StringBuilder $this$lastChar)
{
Intrinsics.checkNotNullParameter($this$lastChar, "");
if ((((CharSequence)$this$lastChar).length() == 0 ? 1 : 0) != 0) {
return "";
}
String str = $this$lastChar.substring($this$lastChar.length() - 1);Intrinsics.checkNotNullExpressionValue(str, "this.substring(length - 1)");return str;
}
可以看到其实也是生成了一个静态方法,java代码调用如下
StringBuilder s = new StringBuilder("value");
String lastChar = EXTKt.getLastChar(s);