【KtAndroid - 8】扩展

图片有部分上传失败……我也是醉了。

这几天在写hook插件,更新可能会断断续续的。后续应该考虑写几篇关于Xposed的……考虑考虑。

1.jpg

函数(方法)的扩展

好了,这些都是后事,黑人兄弟毕竟都还没来,是吧。

今天的主题,函数的扩展

我们先看一段javascript代码:

var Action = function(){};
Action.prototype.sayHello = function(){
    console.log('hello');
}
Action.prototype.leave = function(){
    console.log('goodbye');
}

//run
var a = new Action();
a.sayHello();

如上所示,javascript通过对象名.prototype,可以随意增加属性与方法,kotlin中也有类似的操作,毕竟kotlin的初衷就是为了简便javascript。

之前提过,在kotlin中有一个空类的概念,也就是

class Empty

那么接下来就为这个空类添加扩展方法

///定义
class Empty

/** 方法的扩展 */
fun Empty.extendFun(param:String){……}

如上,一个方法的扩展就好了。

格式就是:fun 类名.扩展方法名

这里的扩展方法可以是任意自定义方法。同时值得注意的是所有的扩展方法都是静态行为,也就是并不会对类与对象造成影响!

这句话不好理解,其实说个例子就明白了

class Empty{
    fun testFun(name: String) {
        println("我是Empty的成员方法:$name")
    }
}

fun Empty.testFun(name: String) {
    println("我是Empty的扩展方法:$name")
}

fun main(args: Array<String>) {
    Empty().testFun("aa")
}
2.png

一. 扩展方法的骚操作

那么既然类的方法可以随意扩展,我们是不是可以为所欲为了。

3.jpg
fun Any?.testNull(): String {
    if(this == null)
        return "我是空的!"

    //有没有this。默认一样,前提是该类中有此方法
    //return toString()
    return this.toString()
}

//run
fun main(args: Array<String>) {
    println(null.testNull())
    println("String".testNull())
    println("哈哈".testNull())
}
4.jpg

注意扩展方法中的

//有没有this。默认一样,前提是该类中有此方法
//return toString()
return this.toString()

千万不要这么用

fun Any?.testNull(): String {
    if(this == null)
        return "我是空的!"

    //有没有this。默认一样,前提是该类中有此方法
    //return toString()
    return testNull()
}
5.jpg

这是递归的表示,如果没有退出条件,那么它就是个永动机。

属性(字段)的扩展

说完了方法的扩展,再来看看属性的扩展

扩展属性允许定义在类或者kotlin文件中,不允许定义在函数中。初始化属性因为属性没有后端字段(backing field),所以不允许被初始化,只能由显式提供的 getter/setter 定义。

嘿嘿,这句话是我抄的。

一切东西都能用代码说事,来!看代码

class Empty1

///看过来
val Empty1.field: Int
    get() = 1


class KtFile1 {
    companion object {
        @JvmStatic
        fun main(args: Array<String>) {
            println(Empty1().field)
        }
    }
}
6.png

说明一下扩展属性的注意事项:

  1. 扩展属性只能用val所修饰

  2. 扩展属性不能声明在方法中,可以是任意类,任意.kt文件

  3. 扩展属性不能直接赋值

来看一下方法中声明扩展属性

7.png

再来看一下直接赋值:

8.png

伴生对象的扩展

之前提过,写在伴生对象内的成员就是静态成员,这也是伴生二字的主要解释了。

既然任何类的方法和字段都能扩展,那么伴生对象是否能被扩展呢?

答案当然是,阔以嘞。

class Empty2 {
    companion object
}

fun Empty2.Companion.extendFun() {
    println("Empty2.Companion.extendFun")
}

//主类
class KtFile1 {
    companion object {
        @JvmStatic
        fun main(args: Array<String>) {
            Empty2.extendFun()
        }
    }
}
9.png

不同包下的类也能使用扩展

10.png

上面说了,任意类都能使用扩展,还举了一个testNull的例子

我现在有两个包:

11.png

现在要做的是,在KtFile.kt下扩展Door<.kt>(后缀名隐藏了)类

12.png

直接导包import即可

作为成员的扩展

这个容易理解,只要把顶层的扩展写到类里就行了

13.png

但是!这是有局限性的,只在该类中有效 在外部任何位置,任何情况都无法直接访问。

最后!!

扩展属性允许定义在类或者kotlin文件中,不允许定义在函数中。初始化属性因为属性没有后端字段(backing field),所以不允许被初始化,只能由显式提供的 getter/setter 定义。

这是我上面抄的一段话,取至菜鸟教程,但是这句话应该是有歧义的,就如菜鸟教程中所说,扩展属性只能用val所修饰,其实不然。

14.png

这样写,编译器也不会报错。

但是,set 方法中是找不到 field 这个默认变量,因此无法赋值!

我想这大概就是菜鸟教程中想要表达的意思。

总结:

  1. 在kotlin中,任意类的成员可以扩展,不论是方法还是属性
  2. 扩展只能在类 或 .kt文件中,不能写在方体内
  3. 属性的扩展不能声明为var,只能声明为val,且初始值只能通过显式的 get 获取
  4. 伴生对象本质上是java中的静态成员,所以伴生对象的扩展就相当于添加静态成员
  5. 不同包下的扩展,可通过import实现导入扩展
  6. 当扩展作为成员时,仅在该范围内有效。(其实是扩展属性只在其声明的.kt文件class中有效,这么说才对。)
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。