七,Kotlin常见高阶函数用法

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了七,Kotlin常见高阶函数用法相关的知识,希望对你有一定的参考价值。

参考技术A 简化版本

kotlin支持对类的方法进行推展,拓展类未定义的方法,拓展本身支持自定义的或者android库中的类

方法覆盖
kotlin支持对局部变量回调的方法实现中支持对回调方法的重定义覆盖,可以使用之前的定义的参数属性(实测也无法重新定义入参,必须使用之前的入参),只需要覆盖方法实现

结果输出.

kotlin支持方法参数定义默认值,也支持定义重载方法,比如自定义View我们一般要实现三个构造方法
kotlin支持方法参数定义默认值,也支持定义重载方法,比如自定义View我们一般要实现三个构造方法

而使用kotlin我们可以给方法定义重载的标记,下面的一个方法和上面的是等价的

kotlin支持泛型的方法处理
可以看下apply和let的实现方式

可以看出这两个方法中都有T.method()的处理,这里可以理解是泛型参数的拓展方法,和上面提到的是一致的。而这里的this指的是拓展方法所指定的泛型对象。
apply这里调用了方法后返回了this,同时泛型返回也是同泛型,也就是说是可以链式调用的,方法内可以使用this.xxx方法获取相应的属性,当然this也是可以省略的,直接也是可以的。

同时apply这里的block是也是一个类拓展函数,指定的是T.()的方法,其实这个就是指定将T作为方法的this上下文进行传递,然后匿名方法体实现中就可以直接通过this获取该对象了
比如我们定义一个匿名方法使用String.(),那么下面的使用是成立的,即使我们并没有指定apply的对象

那么T.()作为匿名方法声明就可以类比上面的,就是把T作为this对象传递到匿名方法体中的。

而let则是把泛型作为it传递的,block: (T)就是传递it上下文的操作了,因为泛型都是T,block方法通过this传递了本身进去,这个就不举例了

那么把上面两个综合起来就可以看这么一个例子

可以看出这里指定了两个泛型T和R, 然后T.(R)这里把T作为this传参,通时R作为it传参到方法体中,也就是这里可以同时使用this和it进行操作
输出结果

那么再加一个参数呢

因为it只支持本身,其实本身也是一种lambda声明,多个参数直接换成多个的格式就可以了,当然this就不用处理了,只需要处理传递进来的两参数即可
输出结果

Kotlin小知识之高阶函数

文章目录

高阶函数

定义高阶函数

  • 高阶函数和Lambda的关系是密不可分的.
  • 像接受Lambda参数的函数就可以称为具有函数式编程风格的API了
  • 当我们想要定义自己的函数式API那就得借助高阶函数来进行实现了.
  • 高阶函数的定义:如果一个函数是另外一个函数,或者一个函数的返回值是另外一个函数,那么就称这个函数为高阶函数.

函数类型

  • 在编程语言当中由整形,布尔类型等字段类型,而Kotlin又增加了一个函数类型的概念

  • 如果我们将这个函数类型添加到一个函数的参数声明或者返回值声明当中,那么这就是高阶函数

  • 定义一个函数类型,不同于定义一个普通的字段类型,函数类型的语法规则如下:

  • (String, Int) -> Unit

  • 既然是定义一个函数类型,那么最关键的就是要声明该函数的参数以及它的返回值是什么.

  • 因此,->左边部分就是用来声明该函数接收什么参数,多个参数之间使用逗号进行隔开,如果不接受任何参使用一对空括号即可.

  • 如果没有返回值就使用Unit,它大致相当于java当中的void

  • 现在将上述函数类型添加到一个函数的参数当中,那这个函数就是一个高阶函数

fun example(func: (String, Int) -> Unit) 
   func("hello", 123) 

  • 在example()函数当中接收了一个函数类型的参数,因此example()函数就是一个高阶函数
  • 而调用一个函数类型的参数,语法类似与调用一个普通的函数,只需要在参数名后面加上一对括号,并在括号当中传入必要的参数
  • 高阶函数允许让函数类型的参数来决定函数的执行逻辑,即使是同一个高阶函数,只要出传入不同的函数类型参数,那么它的执行逻辑和最终返回结果就可能是完全不同的

高阶函数示例

  • 新建一个HigherOrderFunction.kt文件
  • 在其中编写一个高阶函数
fun num1AndNum2(num1: Int, num2: Int, operation: (Int, Int) -> Int) : Int 
    val result = operation(num1, num2)
    return result
 
  • 高阶函数定义好了,该高阶函数接收了一个函数类型的参数,因此我们要去编写所对应的函数类型相匹配的函数即可.
fun plus(num1: Int, num2: Int) : Int 
    return num1 + num2


fun minus(num1: Int, num2: Int) : Int 
    return num1 - num2

  • 这里定义了两个函数,都是和高阶函数中函数类型的参数相匹配的,但是这两个函数却实现的是不同的功能,第一个是俩数之和,第二个是俩数之差.
  • 有了上述的函数之后,我们就可以调用num1AndNum2()函数了,在main函数当中编写如下代码
fun main() 
    val num1 = 5
    val num2 = 1
    val res1 = num1AndNum2(num1, num2, ::plus)
    val res2 = num1AndNum2(num1, num2, ::minus)
    println("res1 is $res1")
    println("res2 is $res2")

  • 这里需要注意的就是调用num1AndNum2()函数的方式,第三个参数使用使用了::plus和::minus的写法,这是一种函数引用的写法,表示将plus()和minus()函数作为参数传递给num1AndNum2()函数
  • 其实我们还可以不编写plus()函数和minus()函数,而是使用Lamdba表达式的方式来调用高阶函数
fun main() 
    val num1 = 5
    val num2 = 1
    val res1 = num1AndNum2(num1, num2)  num1, num2 ->
        num1 + num2
    
    val res2 = num1AndNum2(num1, num2)  num1, num2 ->
        num1 - num2
    
    println("res1 is $res1")
    println("res2 is $res2")

  • 使用高阶函数模拟一个apply函数的类似功能
  • 给StringBuilder类定义一个扩展函数builder(),然后这个扩展函数builder又接收了一个函数类型的参数,这个函数类型的参数它的参数为空,返回值也为空,但是builder扩展函数的返回值为StringBuild
fun StringBuilder.builder(block: StringBuilder.() -> Unit) : StringBuilder 
    block()
    return this

  • 需要注意的就是这个函数类型的声明方式和之前的示例有所不同,它在函数类型参数前面加上了StringBuilder.,这个意思就是在函数类型之前加上ClassName.表示该函数类型的定义在哪个类当中,这个才是定义高阶函数完整的语法规则.
  • 现在我们就可以使用builder函数来代替apply函数完成一些操作
fun main() 
    val list = listOf("Apple", "Banana", "Orange")
    val res = StringBuilder().builder 
        append("Start eating fruits.\\n")
        for (f in list) 
            append(f).append("\\n")
        
        append("Ate all fruits.")
    
    println(res)

内联函数

内联函数的作用

  • 要分析内联函数的作用,得先知道高阶函数的原理
  • 我们首先要知道Kotlin中的代码最后还是要转换称为Java字节码的
  • 但是Java当中又没有对应的高阶函数的概念,其实这一切都要归功于Kotlin的编译器,Kotlin的编译器会将高阶函数的代码转换称为Java当中的语法结构.
  • 转换称为Java语法结构之后,函数类型的参数会被转换成为一个接口,这个接口是Kotlin内置的一个接口,在这个接口当中有一个带实现的invoke()方法
  • 所以最后实际上调用函数类型参数就是调用了接口的invoke()函数,并将相应的参数传递进去即可.
  • 然后之前函数类型参数的Lambda表达式就会改写成为了接口的匿名类实现
  • 原来我们一直使用的Lambda表达式会在底层被转换成为匿名类的实现方式
  • 这就表明我们每调用一次Lambda表达式,都会创建一个新的匿名类实例,这也会造成额外的内存和性能开销
  • 为了解决这个问题,Kotlin提供了内联函数的功能,它可以将Lambda表达式带来的内存和性能开销问题完全消除.

内联函数的用法

  • 内联函数的用法非常简单只需要在定义高阶函数时加上inline关键字的声明即可
inline fun num1AndNum2(num1: Int, num2: Int, operation: (Int, Int) -> Int): Int 
    val result = operation(num1, num2)
    return result

  • 那么内联函数的工作原理其实就是:Kotlin编译器会将内联函数中的代码在编译的时候自动替换到调用它的地方,这样也就不存在运行时的开销了.
  • 大致过程就是:Kotlin编译器会将Lambda表达式中的代码替换到函数类型参数调用的地方

  • 然后再将内联函数中的代码替换到函数调用的地方

  • 最终的代码就成了这个样子

  • 这样内联函数就可以完全消除Lambda表达式带来的内存和性能开销问题

noinline与crossinline

  • 一个比较特殊的情况.比如,一个高阶函数中如果接收了两个或者更多函数类型的参数,这时我们给函数加上了inline关键字,那么Kotlin编译器会自动将所有引用的Lambda表达式全部进行替换.
  • 但是我们现在只想给其中一个Lambad表达式该怎么办,那么就可以使用noline关键字,如下所示:
inline fun inlineTest(block1: () -> Unit, noinline block2: () -> Unit) 
    

  • 为什么Kotlin要提供一个noinline关键字来排除内联功能?因为内联函数类型参数在编译的时候会被进行代码替换,因此他没有真正的参数属性,非内联的函数类型参数可以自由地传递给其他任何函数,因为它就是一个真实参数,而内联函数类型参数只允许传递给另外一个内联函数,这就是它最大的局限性.
  • 另外内敛函数和非内联函数还有一个重要的区别,那就是内联函数所引用的Lambda表达式中时可以使用return关键字进行函数返回.
  • 将高阶函数声明称为内联函数是一种良好的编程习惯,事实上大多数的高阶函数都可以直接声明成为内联函数,但是也是有少部分情况是例外的:
inline fun runRunnable(block: () -> Unit) 
    val ruunable = Runnable 
        block()
    
    runnable.run()

  • 这段代码在没有加上inline关键字的时候肯定是没有问题,但是将这个高阶函数声明成内联函数就会出现下面的问题

  • 上述代码出现问题的主要原因是:在runRunnable()函数中,我们创建了一个Runnable对象,并在Runnable的Lambda表达式中调用了传入的函数类型参数.
  • 而Lambda表达式在编译的时候会转换成为匿名类的实现方式,也就是说,上述代码实际上是在匿名类中调用了传入的函数类型参数.
  • 而内联函数所引用的Lambda表达式允许使用return关键字进行函数返回,但是由于我们是在匿名内部类中调用的函数类型的参数,此时是不可能进行外层调用函数返回的,最多只能对匿名类中的调用函数进行返回
  • 也就是说我们在高阶函数当中创建了另外Lambda或者匿名类的实现,并且在这些实现中调用函数类型参数,此时再将高阶函数声明成内联函数,就一定会提示错误.
  • 所以在这个时候,我们如果还想要使用内联函数的时候,就需要借助crossinline关键字就可以很好的解决问题.
inline fun runRunnable(crossinline block: () -> Unit) 
    val ruunable = Runnable 
        block()
    
    runnable.run()

  • 这个关键字其实就相当于一个契约一样,它用于保证在内敛函数的Lambda表达式一定不会使用return关键字,这样冲突就不存在了.

以上是关于七,Kotlin常见高阶函数用法的主要内容,如果未能解决你的问题,请参考以下文章

Kotlin之高阶函数

Android:Kotlin详细入门学习指南-高阶函数-基础语法

Android:Kotlin详细入门学习指南-高阶函数-基础语法

Kotlin——高级篇:高阶函数详解与标准的高阶函数使用

Kotlin的高阶函数和常用高阶函数

kotlin学习之高阶函数及常用基本高阶函数