Kotlin Lambda 巩固
return num * 2
}
// (Int) -> Int: 表示入参是 Int,返回值是 Int 的函数
fun a(function: (Int) -> Int): String {
return (function(2) + 1).toString()
}
fun main() {
a { b(5) }
}
PS:这样写感觉歧义,但这就是一个高阶函数的写法,大家可以思考一下,输出的结果是 11 还是 5?
答案是 11,Kt 会将 B 函数进行运算,结果传给 A 函数。
因为无论是名字,还是写法,都参照了数学中的高阶函数的用法,大家看下下面数学代数式:
a(x) = b(x) + 1
当然了,函数不仅可以作为参数进行传递,也可以作为返回值进行传递或者对变量进行复制:
fun a(): (Int) -> Unit {
...
}
val d = a
这就是高阶函数的究极意义,除此之外,没有别的特殊功能。
其次根据我们对 Java 的理解,参数都是对象,都是类的具体实例,就算是基本类型,它也会有装箱和拆箱,本质上也是实例对象。
那么 函数
做为参数传递时,它是什么呢?
答案:也是对象。
在编译时期,就已经将其转化成对象进行传递,所以从这个角度来看:
普通函数只是单单调用时,它不是对象,只能说是一个执行任务的逻辑序列
当用函数作为参数进行传递时,该函数就不止是一个逻辑序列,而是一个【对象】。
除了上述的用法外,还可以通过加 双冒号的形式来让函数直接成为一个对象而使用:
fun main() {
// 双冒号,直接带入这个函数
a(::b)
// 在之后可以加括号来表示参数的带入
(::b)(1)
}
通过加双冒号的形式,使得函数成为一个对象,它的本质是使用了对象的 invoke()
的方式去做。
直接更直观的表示了,函数传参不过是传入对象而已。
=========================================================================
我们还可以以匿名函数的形式,来实现上面 a 函数调用 b 函数的方式:
fun a(function: (Int) -> Int): String {
return (function(2) + 1).toString()
}
fun main() {
print(a(
fun(num: Int): Int {
return num * 2
}
))
}
我们将 b 函数直接写进 a 函数中,并且去掉名字 b 就行了。如果写上名字,编译器还会报错,这是因为匿名函数只会在这一处被调用,所以它没有被命名的意义了,所以 Kotlin 索性就不给我们命名了。
来看看 Java 中如何对一个 View 实现监听
// 1. 声明接口回调
public interface OnEndListener{
void onClick(View v);
}
// 2. 设置回调
public void setOnEndListener(OnEndListener listener) {
this.listener = listener;
}
// 3. 设置监听器
view.setOnEndListener(new OnEndListener() {
@Override
void onClick(View v) {
doSomeThing();
}
});
这本质上就是 Java 实现函数式编程的方法-----依靠监听器回调
而 Kotlin 中,因为函数可以做为参数直接传递,那么我们就没必要声明接口了,因为接口只是函数的壳。
OnClickListener
是壳,里面的 onClick
才是我们想要传入的东西。
到了 Kotlin 中,只需要更改 2、3 步:
// 1. 声明接受一个 入参为 View,返回值是 Unit 的函数
fun setOnEndListener(onClick: (View) -> Unit) {
this.onClick = onClick
}
// 设置监听器,传入一个匿名函数 符合 入参为 View,返回值是 Unit 的函数
view.setOnEndListener(fun(v: View): Unit) {
doSomeThing()
})
Lambda 表达式可以简化匿名函数,写成下面的形式:
view.setOnEndListener({ v: View ->
doSomeThing()
})
==============================================================================
Lambda 表达式表示一个函数,如果该函数作为参数,是某个函数调用里的最后一个参数,可以把大括号写在外面。
view.setOnEndListener() { v: View ->
doSomeThing()
}
如果是 Lmabda 函数式唯一参数,那么括号都不用写了:
view.setOnEndListener { v: View ->
doSomeThing()
}
当 Lambda 函数式的参数没有或只有一个(默认为 it)时,连参数都不用写了:
view.setOnEndListener {
doSomeThing()
}
评论