}
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()
})
==============================================================================
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
87683)]
[外链图片转存中…(img-3XdDgVjg-1715676687687)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!