定义:如果一个函数以另一个函数为参数/返回值是一个函数,那么该函数就称为高阶函数。
该定义有些类似于数学中的复合函数,以函数为参数
java里不可以把方法作为参数传递,但可以用另外一个变通方案:接口。
可以通过接口的方式把方法包装起来,然后把这个接口的类型作为外部方法的参数类型,在调用外部方法时,传递接口的对象来作为参数
举个例子 我们在用户发生点击行为时会触发点击事件,所谓点击事件最核心内容就是调用内部的一个OnClickListener的onClick()方法,而所谓OnClickListener只是一个壳,它的核心全在内部那个onClick方法上。我们穿过来一个OnClickListener,其实是传过来一个稍后可以被调用的onClick()方法,只不过因为Java不允许传递方法,所以我们才把它包进了一个对象里来进行传递。
在Kotlin里,函数参数可以是函数类型的,当一个函数含有函数类型的参数时,如果你调用它你就可以传入一个函数类型的对象给它。函数类型不是一个类型,而是一类类型。函数类型可以有各种不同的参数和返回值的类型的搭配,这些搭配属于不同的函数类型。如无参数无返回值和单Int参数String类型返回值不是同一种函数类型。对于函数类型的参数,要指明有几个参数、参数的类型是什 么以及返回值类型是什么。
对于高阶函数,还可以把它赋值给一个变量。对于一个声明好的函数,不管要把它作为参数赋给另一个函数还是赋值给一个变量,都需要在函数名的左边加上双冒号。(::)双冒号叫做函数引用(function reference),加了这两个冒号,函数才变成一个对象(或者说双冒号是一个指向对象的引用)。这个对象不是函数本身,而是一个和这个函数具有相同功能的对象。你可以怎么用函数,就可以怎么用这个加了双冒号的对象。
对象是不能加括号来调用的,但是函数类型的对象可以。这其实是个假调用,是Kotlin的语法糖,实际上你对一个函数类型的对象加括号、加参数,它真正调用的是这个对象的invoke()函数,所以你可以对一个函数类型的对象调用invoke(),却不可以对一个函数这样做。因为只有函数类型的对象有自带这个invoke()函数。
除了加双冒号这种写法,还可以直接把函数挪过来写(这样就不需要函数名了,也就是传说中的匿名函数) 。左边右边都有名字这种写法Kotlin不允许,那既然右边的函数要名字也没有用,那就干脆不要名字了。
正常的话要往类里面添加方法,需要在这个类里面去定义新的方法,即要获取到这个类的读写权限才能够往里面添加新的方法。
弊端是有些类我们没有读写权限,比如Android Studio里面的AppCompatActivity这个类,它是AppCompat这个库给我们的这个类,我们是没有这个类的读写权限的,所以就无法往里面添加新的方法。
此处就引入一个新的概念,拓展函数
拓展函数:可以在没有一个类的读写权限的情况下打开一个类,向类里面添加自定义的方法
拓展函数的定义:fun String.helloworld(){
println("hello world")
}
拓展函数的调用:fun main() {
"1".helloworld()//此时helloworld是String里面的一个实例方法,任何一个String的实例都可调用此方法
}
Java在String类中定义了很多可供我们使用的方法,但有些我们想要的功能它并没有提供,但我们又没有办法后期对它进行添加,而有了拓展方法就可以不断朝String这个类里添加新的功能,就会让它的拓展性变得很强,比如Kotlin的String这个类里就添加了很多Java中所没有的功能,比如capitalize()【返回输入的字符串并且首字母变为大写】,reversed【将输入的字符串顺序颠倒】
fun main() {
println("iloveyou".capitalize())
println("iloveyou".reversed())
}
输出结果:
Iloveyou
uoyevoli
拓展属性:
val String.firstWord3:String
get() {
val index =this.indexOf(" ")
return if (index <0)this else this.substring(0,index)
}
//indexof 如果找不到要找的字符就返回-1
Lambda表达式 java也支持(java8)
Lambda表达式的定义:最直白的语言阐述,就是一小段可以作为参数传递的代码
正常情况下向函数传参数只能传变量,而借助Lambda表达式却可以传一小段代码
Kotlin其实没有对“一小段代码”的代码量进行限制,但通常我们不会写太多以免影响可读性
如果Lambda是函数的最后一个参数,可以把Lambda写在括号的外面,而如果Lambda是函数唯一的参数,还可以把括号去掉,如果Lambda是单参数的,不用这个参数这个参数也可以省略不写。(用了的话也可以不写,因为Kotlin的Lambda对省略的唯一参数有有默认的名字:it)
Lambda这也不写那也不写,他怎么知道自己的参数和返回值类型呢?调用的函数在声明的地方有明确的参数信息,把参数的参数类型和返回值写的清清楚楚,所以Lambda才不用写。
当要把一个匿名函数赋值给变量而不是作为函数参数传递的话,如果也简写成Lambda的形式,就不能省略掉Lambda的参数类型,因为它无法从上下文推断出这个参数的类型。如果出于场景需求或个人偏好,就是想省略参数类型,那需要给左边的变量指明类型。Lambda的返回值不是用return来返回,而是直接取最后一行代码的值。如果写了,会把最后一行代码作为外层函数的返回值来直接结束外层函数,但如果只是想返回Lambda,这样写就出错了。
因为Lambda是个代码块,它总能根据最后一行代码推测出返回值的类型,所以返回值类型确实可以不写。实际上Kotlin的Lambda也是写不了返回值类型的,语法上不支持。
总结:Kotlin的匿名函数不是函数,而是一个对象,一个函数类型的对象,和双冒号加函数名是一个东西,和函数不是。Lambda也是函数类型的对象。