}

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()的方式去做。

直接更直观的表示了,函数传参不过是传入对象而已。

3. 匿名函数

=========================================================================

我们还可以以匿名函数的形式,来实现上面 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()

})

4. Lambda表达式

==============================================================================

结尾

  • 腾讯T4级别Android架构技术脑图;查漏补缺,体系化深入学习提升

img

  • 一线互联网Android面试题含详解(初级到高级专题)

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

img

有Android开发3-5年基础,希望突破瓶颈,成为架构师的小伙伴,可以关注我
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
54930)]

  • 一线互联网Android面试题含详解(初级到高级专题)

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

[外链图片转存中…(img-JJPj6VFV-1714898054934)]

有Android开发3-5年基础,希望突破瓶颈,成为架构师的小伙伴,可以关注我
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

Logo

开源鸿蒙跨平台开发社区汇聚开发者与厂商,共建“一次开发,多端部署”的开源生态,致力于降低跨端开发门槛,推动万物智联创新。

更多推荐