不学点高阶函数,如何愉快的装逼!

如果你开始接触函数式编程,你一定听说过高阶函数。在维基百科它的中文解释是这样的:

在数学和计算机科学中,高阶函数是至少满足下列一个条件的函数:

  • 接受一个或多个函数作为输入
  • 输出一个函数

看起它就是ObjC语言中入参或者返回值为block的block或者函数,在Swift语言中即为入参或者返回值为函数的函数。那它们在实际的开发过程中究竟起着什么样的作用呢?我们将从入参、返回值和综合使用三部分来看这个问题:

函数作为入参

函数作为入参似乎无论在ObjC时代还是Swift时代都是司空见惯的事情,例如AFNetworking就用两个入参block分别回调成功与失败。Swift中更是加了一个尾闭包的语法(最后一个参数为函数,可以不写括号或者写到括号外面直接跟随方法名),例如下面这样:


  1. [1, 2, 3].forEach { item in 
  2.     print(item) 
  3. }  

我们可以将入参为函数的函数分为两类,escaping函数入参和noescape函数入参,区别在于这个入参的函数是在执行过程内被调用还是在执行过程外被调用。执行过程外被调用的一般用于callback用途,例如:


  1. Alamofire.request("https://httpbin.org/get").responseJSON { response in 
  2.     print(response.request)  // original URL request   
  3.     print(response.response) // HTTP URL response   
  4.     print(response.data)     // server data   
  5.     print(response.result)   // result of response serialization   
  6.  
  7.     if let JSON = response.result.value {        print("JSON: \(JSON)") 
  8.     } 
  9. }  

这个response的入参函数就作为网络请求回来的一个callback,并不会在执行responseJSON这个函数的时候被调用。另外我们来观察forEach的代码,可以推断入参的函数一定会在forEach执行过程中使用,执行完就没有利用意义,这类就是noescape函数。

callback的用法大家应该比较熟悉了,介绍给大家noescape入参的一些用法:

1. 自由构造器

看过GoF设计模式的同学不知道是否还记得构造器模式,Android中的构造器模式类似如下:


  1. new AlertDialog.Builder(this) 
  2.   .setIcon(R.drawable.find_daycycle_icon) 
  3.   .setTitle("提醒") 
  4.   .create() 
  5.   .show();  
  1. 构造一个对象需要很多的参数
  2. 这些参数里面很多有默认值
  3. 这些参数对应的属性未来不希望被修改

那么用这样的模式就可以直观又精巧的展示构建过程。

如果使用noescape入参函数还可以更简单的构造出这种代码,只需要传入一个入参为builder的对象就可以了,如下:


  1. // 实现在这里  class SomeBuilder {    var prop1: Int 
  2.     var prop2: Bool 
  3.     var prop3: String 
  4.     init() {        // default value   
  5.         prop1 = 0 
  6.         prop2 = true 
  7.         prop3 = "some string" 
  8.     } 
  9. }class SomeObj { 
  10.     private var prop1: Int 
  11.     private var prop2: Bool 
  12.     private var prop3: String 
  13.     init(_ builderBlock:(SomeBuilder) -> Void) {        let someBuilder = SomeBuilder() 
  14.         builderBlock(someBuilder) // noescape 入参的使用   
  15.         prop1 = someBuilder.prop1 
  16.         prop2 = someBuilder.prop2 
  17.         prop3 = someBuilder.prop3 
  18.     } 
  19. }// 使用的时候  let someOjb = SomeObj { builder in 
  20.     builder.prop1 = 15 
  21.     builder.prop2 = false 
  22.     builder.prop3 = "haha"}  

2. 自动配对操作

很多时候,我们开发过程中都会遇到必须配对才能正常工作的API,例如打开文件和关闭文件、进入edit模式退出edit模式等。虽然swift语言给我们defer这样的语法糖避免大家忘记配对操作,但是代码看起来还是不那么顺眼


  1. func updateTableView1() {    self.tableView.beginUpdates()    self.tableView.insertRows(at: [IndexPath(row: 2, section: 0)], with: .fade)    self.tableView.deleteRows(at: [IndexPath(row: 5, section: 0)], with: .fade)  self.tableView.endUpdates() // 容易漏掉或者上面出现异常  }func updateTableView2() {    self.tableView.beginUpdates() 
  2.     defer {        self.tableView.endUpdates() 
  3.     }    self.tableView.insertRows(at: [IndexPath(row: 2, section: 0)], with: .fade)    self.tableView.deleteRows(at: [IndexPath(row: 5, section: 0)], with: .fade) 

利用noescape入参,我们可以将要操作的过程封装起来,使得上层看起来更规整


  1. // 扩展一下UITableView  extension UITableView {    func updateCells(updateBlock: (UITableView) -> Void) { 
  2.         beginUpdates() 
  3.         defer { 
  4.             endUpdates() 
  5.         } 
  6.         updateBlock(self) 
  7.     } 
  8. }func updateTableView() {  // 使用的时候   
  9.     self.tableView.updateCells { (tableView) in 
  10.         tableView.insertRows(at: [IndexPath(row: 2, section: 0)], with: .fade) 
  11.         tableView.deleteRows(at: [IndexPath(row: 5, section: 0)], with: .fade) 
  12.     } 

函数作为入参就简单介绍到这里,下面看看函数作为返回值。

函数作为返回值

在大家的日常开发中,函数作为返回值的情况想必是少之又少。不过,如果能简单利用起来,就会让代码一下子清爽很多。

首先没有争议的就是我们有很多的API都是需要函数作为入参的,无论是上一节提到过的escaping入参还是noescape入参。所以很多的时候,大家写的代码重复率会很高,例如:


  1. let array = [1, 3, 55, 47, 92, 77, 801]let array1 = array.filter { $0 > 3 * 3}let array2 = array.filter { $0 > 4 * 4}let array3 = array.filter { $0 > 2 * 2}let array4 = array.filter { $0 > 5 * 5} 

一段从数组中找到大于某个数平方的代码,如果不封装,看起来应该是这样的。为了简化,通常会封装成如下的两个样子:


  1. func biggerThanPowWith(array: [Int], value: Int) -> [Int] { 
  2.     return array.filter { $0 > value * value} 
  3.  
  4. let array1 = biggerThanPowWith(array: array, value: 3) 
  5. let array2 = biggerThanPowWith(array: array, value: 4) 
  6. let array3 = biggerThanPowWith(array: array, value: 2) 
  7. let array4 = biggerThanPowWith(array: array, value: 5) 

如果用高阶函数的返回值函数,可以做成这样一个高阶函数:


  1. // 一个返回(Int)->Bool的函数  func biggerThanPow2With(value: Int) -> (Int) -> Bool {    return { $0 > value * value } 
  2. }let array1 = array.filter(biggerThanPow2With(value: 3))let array2 = array.filter(biggerThanPow2With(value: 4))let array3 = array.filter(biggerThanPow2With(value: 2))let array4 = array.filter(biggerThanPow2With(value: 5)) 

你一定会说,两者看起来没啥区别。所以这里面需要讲一下使用高阶返回函数的几点好处

1. 不需要wrapper函数也不需要打开原始类

如同上面的简单封装,其实就是一个wrapper函数,把array作为入参带入进来。这样写代码和看代码的时候就稍微不爽一点,毕竟大家都喜欢OOP嘛。如果要OOP,那就势必要对原始类进行扩展,一种方式是加extension,或者直接给类加一个新的方法。

2. 阅读代码的时候一目了然

使用简单封装的时候,看代码的人并不知道内部使用了filter这个函数,必须要查看源码才能知道。但是用高阶函数的时候,一下子就知道了使用了系统库的filter。

3. 更容易复用

这也是最关键的一点,更细粒度的高阶函数,可以更方便的复用,例如我们知道Set也是有filter这个方法的,复用起来就这样:


  1. let set = Set<Int>(arrayLiteral: 1, 3, 7, 9, 17, 55, 47, 92, 77, 801)let set1 = set.filter(biggerThanPow2With(value: 3))let set2 = set.filter(biggerThanPow2With(value: 9)) 

回忆下上面的简单封装,是不是就无法重用了呢?

类似的返回函数的高阶函数还可以有很多例子,例如上面说过的builder,假如每次都需要定制成特殊的样子,但是某个字段不同,就可以用高阶函数很容易打造出来:


  1. func builerWithDifferentProp3(prop3: String) -> (SomeBuilder) -> Void {    return { builder in 
  2.         builder.prop1 = 15 
  3.         builder.prop2 = true 
  4.         builder.prop3 = prop3 
  5.     } 
  6. }let someObj1 = SomeObj.init(builerWithDifferentProp3(prop3: "a"))let someObj2 = SomeObj.init(builerWithDifferentProp3(prop3: "b"))let someObj3 = SomeObj.init(builerWithDifferentProp3(prop3: "c")) 

介绍完入参与返回值,还有另外的一个组合模式,那就是入参是一个函数,返回值也是一个函数的情况,我们来看看这种情况。

入参函数 && 返回值函数

这样的一个函数看起来会很恐怖,swift会声明成:


  1. func someFunc<A, B, C, D>(_ a: (A) -> B)-> (C) -> D 

objective-c会声明成


  1. - (id (^)(id))someFunc:(id (^)(id))block 

让我们先从一个小的例子来讲起,回忆一下我们刚刚做的biggerThanPow2With这个函数,如果我们要一个notBiggerThanPow2With怎么办呢?你知道我一定不会说再写一个。所以我告诉你我会这样写:


  1. func not<T>(_ origin_func: @escaping (T) -> Bool) -> (T) -> Bool {    return { !origin_func($0) } 
  2. }let array5 = array.filter(not(biggerThanPow2With(value: 9))) 

并不需要一个notBiggerThanPow2With函数,我们只需要实现一个not就可以了。它的入参是一个(T) -> Bool,返回值也是(T) -> Bool,只需要在执行block内部的时候用个取反就可以了。这样不单可以解决刚才的问题,还可以解决任何(T) -> Bool类型函数的取反问题,比如我们有一个odd(_: int)方法来过滤奇数,那我们就可以用even=not(odd)得到一个过滤偶数的函数了。


  1. func odd(_ value: Int) -> Bool {    return value % 2 == 1}let array6 = array.filter(odd)let array7 = array.filter(not(odd))let even = not(odd)let array8 = array.filter(even) 

大家可以看下上面的biggerThanPow2With时我们讨论过的,如果biggerThanPow2With不是一个返回函数的高阶函数,那它就不太容易用not函数来加工了。

综上,如果一个入参和返回值都是函数的函数就是这样的一个转换函数,它能够让我们用更少的代码组合出更多的函数。另外需要注意一下,如果返回的函数里面闭包了入参的函数,那么入参函数就是escaping入参了。

下面再展示给大家两个函数,一个交换参数的函数exchangeParam,另一个是柯里化函数currying:


  1. func exchangeParam<A, B, C>(_ block: @escaping (A, B) -> C) -> (B, A) -> C {    return { block($1, $0) } 
  2. }func currying<A, B, C>(_ block: @escaping (A, B) -> C, _ value: A) -> (B) -> C {    return { block(value, $0) } 

swift语言里面>是一个入参(a, b)的函数,所以>(5, 3) == true。我们使用exchangeParam交换参数就变成了(b, a),这时exchangeParam(>)(5, 3)就等于false了。

而currying函数又把参数b固定为一个常量9,所以currying(exchangeParam(>), 9)就是大于9的函数意思。

这个例子里就利用了全部的预制函数和通用函数,没有借助任何的命令与业务函数声明实现了一个从数组中过滤大于9的子数组的需求。试想一下,如果我们更多的使用这样的高阶函数,代码中是不是很多的逻辑可以更容易的互相组合,而这就是函数式编程的魅力。

总结

高阶函数的引入,无论是从函数式编程还是从非函数式编程都带给我们代码一定程度的简化,使得我们的API更加简易可用,复用更充分。然而本文的例子不过是冰山一角,更多的内容还需要大家的不断尝试和创新,也可以通过学习更多的函数式编程范式来加深理解。

作者介绍:

作者:臧成威

来源:51CTO

时间: 2024-09-09 12:24:02

不学点高阶函数,如何愉快的装逼!的相关文章

函数式接口、默认方法、纯函数、函数的副作用、高阶函数、可变的和不可变的、函数式编程和 Lambda 表达式 - 响应式编程 [Android RxJava2](这到底是什么)第三部分

本文讲的是函数式接口.默认方法.纯函数.函数的副作用.高阶函数.可变的和不可变的.函数式编程和 Lambda 表达式 - 响应式编程 [Android RxJava2](这到底是什么)第三部分, 太棒了,我们又来到新的一天.这一次,我们要学一些新的东西让今天变得有意思起来. 大家好,希望你们都过得不错.这是我们的 RxJava2 Android 系列的第三篇文章. 第一部分 第二部分 在这篇文章中,我们将讨论函数式的接口,函数式编程,Lambda 表达式以及与 Java 8 的相关的其它内容.这

F#教程:高阶函数

所谓高阶函数就是将某个函数作为输入参数或者返回值的函数.从名字上来看挺难理解的,不过从C#的角度来看就是传入或返回delegate之类的. 在我们自己定义高阶函数之前我们还是先学会使用高阶函数. List中定义了很多高阶函数,这回就学习下其中的几个.首先试下find函数. let list = [15; 7; 8; 3; 6; 10] let even n = n % 2 = 0 let x = List.find even list printfn "%A" x 其中,find的第一

F#教程-定义高阶函数

目前为止我们已经尝试写了些高阶函数代码.这回我们学着定义如下高阶函数: f(g,a) = g(g(a)) 该函数在得到函数g和传入g的参数a后计算g(a),并将结果作为再次调用函数g的参数.语言描述确实很复杂. F#表示如下: let f g a = g (g a) 使用Pipeline就可以改写成: let f g a = g a |> g C#的话可能会像这样: public T f<T>(Func<T,T> g, T a) { return g(g(a)); } F#就

高阶函数、委托与匿名方法

高阶函数(higher-order function)是指把另一个函数作为参数或返回值的函数.例如 在JavaScript语言中,Function是顶级类型.一个函数就是类型为 Function的顶级对象,自 然就可以作为另一个函数的参数或返回值.例如在Microsoft AJAX Library(ASP.NET AJAX 的客户端类库)中有一个被广泛使用的createDelegate方法.该方法接受一个对象A和一个函 数F作为参数,并返回一个函数R.当调用函 数R时,F函数将被调用,并且保证无

Javascript 高阶函数使用介绍

  高阶函数(higher-order function)-如果一个函数接收的参数为或返回的值为函数,那么我们可以将这个函数称为高阶函数.众所周知,JavaScript是一种弱类型的语言:JavaScript的函数既不对输入的参数,也不对函数的输出值作强定义和类型检查,那么函数可以成为参数,也可以成为输出值,这就体现了JavaScript对高阶函数的原生支持. 一.参数为函数的高阶函数: ? 1 2 3 4 5 6 function funcTest(f){ //简易判断一下实参是否为函数 if

ES6中的高阶函数:如同 a =&gt; b =&gt; c 一样简单

作者:Sequoia McDowell 2016年01月16日 ES6来啦!随着越来越多的代码库和思潮引领者开始在他们的代码中使用ES6,以往被认为是"仅需了解"的ES6特性变成了必需的代码常识.这不仅仅是新的语法学习 - 在许多范例中, ES6中新的语言特性可以让在ES5中写起来非常麻烦的表达变得更加简单,进而鼓励了新表达方式的使用.下面我们将关注一个这样简洁表达的使用范例:ES6中的箭头函数如何使高阶函数的书写更加简便. 高阶函数是至少具有以下两种功能之一的函数: 使用一个或多个函

Coursera Scala 5-4:List的高阶函数

Coursera Scala 5-4:List的高阶函数 Recurring Patterns for Computations on Lists 重复出现的Lists计算模式 lists的很多函数有相似的结构,重复出现的模式有: 用某个方法转换每个元素 用某个条件提取元素 用某种方法链接元素 函数式编程语言,让程序员能写出更通用的计算模式,通过使用高阶函数. Applying a Function to Elements of a List 将一个list的所有元素,进行转换.例子:返回一个新

Scala入门到精通——第十三节 高阶函数

本节主要内容 高阶函数简介 Scala中的常用高阶函数 SAM转换 函数柯里化 部分应用函数 1. 高阶函数简介 高阶函数主要有两种:一种是将一个函数当做另外一个函数的参数(即函数参数):另外一种是返回值是函数的函数.这两种在本教程的第五节 函数与闭包中已经有所涉及,这里简单地回顾一下: (1)函数参数 //函数参数,即传入另一个函数的参数是函数 //((Int)=>String)=>String scala> def convertIntToString(f:(Int)=>Str

Kotlin 简单优雅的高阶函数

Kotlin 简单优雅的高阶函数 最新上架!!!< Kotlin极简教程> 陈光剑 (机械工业出版社) 可直接打开京东,淘宝,当当===> 搜索: Kotlin 极简教程http://www.jianshu.com/p/35b487734339 函数代表一种关系 f 的蕴涵逻辑流.这种蕴涵逻辑流,其实就是映射(Mapping). 一切皆是映射. 我们说组合是编程的本质,其实,组合就是建立映射关系. 我们说, 程序 = 算法+数据结构 我们把程序看做图论里面的一张图G,这里的数据结构就是图