函数式编程的思想主要是把函数(一小段的“行为”)当作值使用,因此,函数也可以被变量保存,当然也可以作为参数传递或者返回值。由于函数具有对于相同输入产生相同输出的性质,因此函数式编程往往意味着简洁,并且便于测试。非常不幸的是,Java 7之前都不支持函数式编程,为了兼容性问题,我们需要在Kotlin上实现,利用Kotlin代替Java开发应该是思想上的转变,而不是纠结于一些语法糖。那么我们首先看一下kotlin这种带一点函数式编程味道的语言是怎么做的。
lambda是什么?
嗯,没错,lambda是什么,lambda一般指的是可以传递到其他函数的一段代码。
由于函数式编程的思想,函数可以当作值使用,也可以被变量保存,也可以作为参数传递或者作为返回值
那么问题就是,在Java/Kotlin中,变量都是有它们的类型的(比如Object),那么函数有类型吗?
显然是有的,一个返回A类对象的函数显然和返回B类对象的函数不是一个类型,一个接收N个参数的函数和一个无参函数显然也不是一个类型,
一个变量也不可能无限制地保存任何类型的函数。
而定义了函数类型的,就是lambda表达式,根据以上的描述,lambda需要表达的函数信息有:
(1)函数的参数描述
(2)函数的返回值
那么lambda的语法就不难理解了:
{x : Int,y : Int -> x + y }
->前面的部分表面此函数接收两个Int作为参数,->后边的部分的最后一条语句决定了此函数的返回值
注意:最后一条语句,以下表达式返回的是x*3
{x : Int,y : Int -> x+33;println("wtf");x+y*77;x*3}
从Q1代码来看,函数的确是有类型区别的:),func1变量仅仅保存以两个Int作为参数并且返回Int值的函数
使用lambda表达式
maxBy是集合的常见函数,用于在一个集合中找到具有最大指定值的对象,它的定义如下:
以函数类型作为参数的函数称为高阶函数,从定义中可以看出,它接受一个函数类型参数,此函数接收一个泛型T作为参数并返回R类型,然后对返回值进行比较,找到最大的对象
我们使用一个lambda作为参数,可以轻松地对我们感兴趣的属性(当然是money啦)进行比较
但是这段语句还是显得啰嗦,事实上我们已经知道countryList肯定存储的是Country对象,根据kotlin的智能转换,我们可以省略参数的类型
另外,如果lambda表达式是函数调用的最后一个实参,它可以放到括号的外边
还有,如果这个lambda只有一个参数的话,你还可以进一步的省略lambda的参数部分,然后用it代替它
虽然确实看起来简便多了,但是在嵌套lambda的时候最好不要这样用,不然别人看你代码时会打死你,尽量还是显式声明每个lambda参数吧
使用Java的函数式接口
lambda很好很强大,但是,Java里可没有高阶函数,那么lambda在这种情形下可以使用吗?
OnClickListener是一个单抽象方法接口(SAM接口、函数式接口),它只拥有一个方法,接受一个View参数并且无返回值
很明显,此接口的参数类型和返回值都已经很明确了,因此在Java8和Kotlin中,我们可以使用lambda表达式简化代码
因此在Java的函数式接口中使用lambda的条件很苛刻,必须是单抽象方法(Single Abstract Method)(这样才能确定参数及返回值)
那么反过来呢?Kotlin与Java之间不是具有100%可交互性吗,那我们在Java代码中是如何使用Kotlin中的lambda语句的呢?kotlin的lambda表达式又是怎么转换为Java代码的呢?当你自己开始思考这个问题的时候,你会对lambda的负面特性有更深的理解。例如,前文所示的接受两个Int并返回Int的函数:{x:Int,y:Int -> x+y},实际上在Java里它会这样"自动实现*
这意味着在Java中调用Kotlin代码时,Java虚拟机为每一句lambda表达式编译一个匿名类,更严重的是,如果lambda还捕捉了其他变量,那么这些变量也会在匿名类中产生相应的字段,并且每次调用都会产生这个匿名类的新实例。
与匿名类不同的是,匿名类要求其参数类型是final的,而lambda无此限制(实际上它还为参数创建了一个包裹对象),因此造成了以上的额外开销。因此,lambda表达式虽然方便,也不要无脑使用,至少不要在循环里面用。再或者,你可以采取一些其他手段来解决这个问题......