Kotlin Lambda巩固
}fun a(function: Fun): String{return function(2).toString()}main() {a(b)}在这个例子中: 是一个类,代表传递了的一个函数,类似Java中的 ,像是万能类。但实际上,在Kotlin中,并没有Fun这样的、或者相似的类,来做到用一个万能类来表示任何一个函数式。而Java中,虽然有 这样的类可以进行转化,但我们平时使用的时候还是会有
}
fun a(function: Fun): String{
return function(2).toString()
}
main() {
a(b)
}
在这个例子中:Fun 是一个类,代表传递了的一个函数,类似Java中的 Class,像是万能类。
但实际上,在Kotlin中,并没有Fun这样的、或者相似的类,来做到用一个万能类来表示任何一个函数式。
而Java中,虽然有 Object这样的类可以进行转化,但我们平时使用的时候还是会有各种各样的转化坑(代码也很挫),所以Kotlin并没有没采取这种做法。
Kotlin借鉴了Python等语言方式,以 “(参数) -> 返回值” 的代码的形式,来描述一个函数,达到 Fun的效果:
fun b(num: Int): Int {
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()
})
==============================================================================
结尾
- 腾讯T4级别Android架构技术脑图;查漏补缺,体系化深入学习提升

- 一线互联网Android面试题含详解(初级到高级专题)
这些题目是今年群友去腾讯、百度、小米、乐视、美团、58、猎豹、360、新浪、搜狐等一线互联网公司面试被问到的题目。并且大多数都整理了答案,熟悉这些知识点会大大增加通过前两轮技术面试的几率

有Android开发3-5年基础,希望突破瓶颈,成为架构师的小伙伴,可以关注我
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
54930)]
- 一线互联网Android面试题含详解(初级到高级专题)
这些题目是今年群友去腾讯、百度、小米、乐视、美团、58、猎豹、360、新浪、搜狐等一线互联网公司面试被问到的题目。并且大多数都整理了答案,熟悉这些知识点会大大增加通过前两轮技术面试的几率
[外链图片转存中…(img-JJPj6VFV-1714898054934)]
有Android开发3-5年基础,希望突破瓶颈,成为架构师的小伙伴,可以关注我
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
更多推荐


所有评论(0)