美文网首页Kotlin从入门到放弃
Kotlin中的函数式编程(1):lambda表达式

Kotlin中的函数式编程(1):lambda表达式

作者: 黑心石 | 来源:发表于2017-10-29 20:10 被阅读0次

        函数式编程的思想主要是把函数(一小段的“行为”)当作值使用,因此,函数也可以被变量保存,当然也可以作为参数传递或者返回值。由于函数具有对于相同输入产生相同输出的性质,因此函数式编程往往意味着简洁,并且便于测试。非常不幸的是,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:函数类型

    从Q1代码来看,函数的确是有类型区别的:),func1变量仅仅保存以两个Int作为参数并且返回Int值的函数

    使用lambda表达式

    maxBy是集合的常见函数,用于在一个集合中找到具有最大指定值的对象,它的定义如下:

    maxBy的源代码

    以函数类型作为参数的函数称为高阶函数,从定义中可以看出,它接受一个函数类型参数,此函数接收一个泛型T作为参数并返回R类型,然后对返回值进行比较,找到最大的对象

    Q2:使用lambda表达式作为参数

    我们使用一个lambda作为参数,可以轻松地对我们感兴趣的属性(当然是money啦)进行比较

    但是这段语句还是显得啰嗦,事实上我们已经知道countryList肯定存储的是Country对象,根据kotlin的智能转换,我们可以省略参数的类型

    另外,如果lambda表达式是函数调用的最后一个实参,它可以放到括号的外边

    还有,如果这个lambda只有一个参数的话,你还可以进一步的省略lambda的参数部分,然后用it代替它

    虽然确实看起来简便多了,但是在嵌套lambda的时候最好不要这样用,不然别人看你代码时会打死你,尽量还是显式声明每个lambda参数吧

    说实话,我还是喜欢简单点

    使用Java的函数式接口

    lambda很好很强大,但是,Java里可没有高阶函数,那么lambda在这种情形下可以使用吗?

    为Button绑定监听器

    OnClickListener是一个单抽象方法接口(SAM接口、函数式接口),它只拥有一个方法,接受一个View参数并且无返回值

    很明显,此接口的参数类型和返回值都已经很明确了,因此在Java8和Kotlin中,我们可以使用lambda表达式简化代码

    lambda表达式与SAM接口

    因此在Java的函数式接口中使用lambda的条件很苛刻,必须是单抽象方法(Single Abstract Method)(这样才能确定参数及返回值)

    那么反过来呢?Kotlin与Java之间不是具有100%可交互性吗,那我们在Java代码中是如何使用Kotlin中的lambda语句的呢?kotlin的lambda表达式又是怎么转换为Java代码的呢?当你自己开始思考这个问题的时候,你会对lambda的负面特性有更深的理解。例如,前文所示的接受两个Int并返回Int的函数:{x:Int,y:Int -> x+y},实际上在Java里它会这样"自动实现*

    根据lambda表达式生成接口和新的类

    这意味着在Java中调用Kotlin代码时,Java虚拟机为每一句lambda表达式编译一个匿名类,更严重的是,如果lambda还捕捉了其他变量,那么这些变量也会在匿名类中产生相应的字段,并且每次调用都会产生这个匿名类的新实例

    捕捉了其他变量的lambda:根据不同的time值生成新的对象

    与匿名类不同的是,匿名类要求其参数类型是final的,而lambda无此限制(实际上它还为参数创建了一个包裹对象),因此造成了以上的额外开销。因此,lambda表达式虽然方便,也不要无脑使用,至少不要在循环里面用。再或者,你可以采取一些其他手段来解决这个问题......

    相关文章

      网友评论

        本文标题:Kotlin中的函数式编程(1):lambda表达式

        本文链接:https://www.haomeiwen.com/subject/dafypxtx.html