Kotlin的高阶函数和常用高阶函数
高阶函数的定义
将函数当做参数或者是返回值的函数
什么是高阶函数
可以看看我们常用的 函数:
首先我们可以知道, 是 的扩展函数,然后参数是 ,但是 不再像和我们以前Java那样传递的是一个对象,这时传递的是一个函数。这个函数的入参为 ,返回值为 。所以 也是一个高阶函数,因为它将函数当做参数进行传递了。我们尝试着去调用一下 函数:
调用的时候,我们将 函数传递给了 函数,这里采用的是函数引用。就上诉代码,我们还可以结合 表达式来进行处理:
其实以上几种的方式得到的结果都是一样的,但是第一种就是简洁了许多。
我们再定义一个类,用来打印 的值:
根据函数引用的特性,我们可以这样调用 函数:。由于 中的 函数的入参类型是 类型,也就是任意类型,不管 传递的是什么值都可以接收。那现在我们再将其作为 的参数传递进去:
编译器告诉我们这个是错误的。那我们来分析一下吧:我们再定义一个 类
然后进行以下操作:
可以看到 分别对 和 的解释:
是一个方法,然后参数类型为 ,返回值为
也是一个方法,但是参数有两个,分别是 和 类型, 返回值为
在 中,只有一个参数传递,但是 需要的是两个参数,肯定就会报错,所以我们需要对此进行修改:
这个样子就OK了。
常用高阶函数
map:变换
通常我们会使用以下的方式来实现对集合中的元素进行修改的操作:
如果采用这种方式,远远不能体现Kotlin的优势了,这个和Java有什么区别呢?「狗子,上map」:
从打印结果可以看到他们的实现效果是一模一样的,这个就是 的功能,可以对集合中的元素进行你想要的操作,是不是跟 的 很类似呢!我们来细看一下 的实现原理:
方法中主要做的就是调用 方法,然后传递的是新创建并且初始长度为10的 和 函数,在 方法中,对集合进行迭代,然后将进行变换后的数据添加到新的集合中,最后返回新的集合。
操作不仅可以将元素变换成与之前类型相同的元素,也可以变化成与之前元素类型不同的元素,具体你想变换成什么类型,这个是不做限制的。
flatMap:变换
看起来跟 很相似,其实真的很类似,搞得有时候自己都不知道应该使用哪个操作符了,那就从源码来看看它们之间的区别吧。
可以看到 中的参数 是一个返回值为 的函数,而 的参数 是一个返回值为 的函数。然后调用 方法,将 和一个新创建的 传递给了 方法。在 方法中,对当前的集合进行了迭代,然后将执行过变换操作后的集合数据全部添加到新的集合中,最终返回新的集合。
和 的主要区别就是在于传入的函数的返回值,一个是任意对象,一个是实现了 接口的对象
reduce
例子:打印集合中的元素之和
还是直接对源码进行分析吧,感觉看了源码就一目了然了。
首先对当前的集合进行判空处理,接着将第一个元素赋值给 , 的类型是 。然后对当前集合进行迭代处理,并调用我们传递进去的参数 , 函数中传递了两个参数,一个是 类型的,一个是集合元素类型的。 函数的返回值也是 类型的,将 的返回值重新赋值给 。迭代完毕以后返回我们的 。
其实通过我们解读源码以后,我们就可以知道 函数会将上一次的计算结果传递到下一次计算中,我们可以利用这个方式来实现以下字符串拼接,当然我们的字符串拼接有其他更好的方式,这里只是做为讲解 的例子而已:
fold:能够添加初始值的reduce
不得不说, 跟 的作用基本是一致的,只是 能够添加初始值,什么叫做能够添加初始值呢?让我们来举个栗子看看呗!
还是看源码吧:
看着源码就会觉得这些函数的操作很是简单了。 函数还有很多的兄弟:
说的再多也不如看结果
这个函数就是多了一个 的参数,具体的用处暂时没有发现,就不做数据打印了。
filter:过滤
例子:过滤集合中的奇数
上源码:
在 中创建新的集合 ,将 和 函数一并传递给 函数。在 函数中,先对当前的集合进行遍历,如果满足条件 就将当前的元素添加到新的集合中, 就是我们传递进来的那个函数,返回值是一个 类型的。
takeWhile:截取集合中的数据,直到第一个不满足条件的元素为止
例子:截取集合中不能够被5整除的数,直到第一个不满足条件的元素为止。
源码:
let:将调用者当做参数传递给指定的函数
例子:省略if空判断
源码:
源码也是很简单,就是将调用者传递给传入进来的函数并执行传入进来的函数。
apply:执行指定函数,并且返回调用者
例子:修改person类的age属性
上诉的例子真的只是例子,它没有跟你讲 有多强大,它只是描述了的作用。源码:
就是单纯的执行函数 并返回调用者。
with
例子:文件读取
源码:
接收两个参数,一个是 ,上诉例子中就是 ,另一个就是 ,在上诉例子中,我们使用了 表达式,所以这个函数就移到了括号外面了。
疑惑点
函数定义中 `T.() -> Unit` 和 `() -> Unit` 的区别
我们一般定义函数都会选择定义
上诉代码表示: 的扩展方法 接收一个 的函数,该函数是无参无返回的。
那我们再见识见识 这种方式定义的方法,其实也很多见了:
可以看到 函数使用的是 这种方式,他们两到底有啥区别呢?
我在 定义了两个方法:
分别使用 调用这两个方法试试:
从图片中可以看出 : 中的 对象指的是调用对象 ,也就是 ,而 没有提示,那么我们就看打印吧:
原来 方法中的 对象指的是外部对象,在这里就是我们的 对象。
总结
这两个函数唯一的区别就是 与 的区别,我们调用时,在代码块里面写this,的时候,根据代码提示,我们可以看到,连个this代表的含义不一样, 里的this代表的是自身实例,而 里,this代表的是外部类的实例
感谢
[Kotlin中,函数作为参数,T.()->Unit 和 ()->Unit 的区别][https://www.jianshu.com/p/88a656e59c61]
This is All