Swift 柯里化
前言
- 由于柯里化在业务层的应用较少,所以从 Swift 3.0 开始移除了柯里化的用法,但是 Swift 的很多底层特性是使用柯里化来表达的。
1、柯里化
1.1 柯里化简介
柯里化(Currying),又称部分求值(Partial Evaluation),是一种函数式编程思想,就是把接受多个参数的函数转换成接收一个单一参数(最初函数的第一个参数)的函数,并且返回一个接受余下参数的新函数技术。
uncurried: 普通函数
// 接收多个参数的函数(与类相关的函数,统称为方法,但是这里就直接说函数了,方便理解)
func add(a: Int, b: Int, c: Int) -> Int { print("\(a) + \(b) + \(c)")
return a + b + c
}
curried: 柯里化函数
// 柯里化函数,Swift 3.0 之前支持这样的语法,可以直接写
func addCur(a: Int)(b: Int)(c: Int) -> Int { println("\(a) + \(b) + \(c)")
return a + b + c
}
1.2 如何定义柯里化函数
定义柯里化函数
func function name (parameters)(parameters) -> return type { statements
}
1.3 柯里化函数实现原理
uncurried: 普通函数
class Currying { // 接收多个参数的函数
func add(a: Int, b: Int, c: Int) -> Int { print("\(a) + \(b) + \(c)")
return a + b + c
}
}
系统自带的柯里化函数
class Currying { func addCur(a: Int)(b: Int)(c: Int) -> Int { print("\(a) + \(b) + \(c)")
return a + b + c
}
}
手动实现柯里化函数
把上面的函数转换为柯里化函数,首先转成接收第一个参数
a,并且返回接收余下第一个参数b的新函数(采用闭包).这里为什么能使用参数
a、b、c?- 利用闭包的值捕获特性,即使这些值作用域不在了,也可以捕获到他们的值。
- 闭包会自动判断捕获的值是值拷贝还是值引用,如果修改了,就是值引用,否则值拷贝。
- 注意只有在闭包中才可以,
a、b、c都在闭包中。
class Currying { // (a: Int) : 参数
// (b: Int) -> (c: Int) -> Int : 函数返回值(一个接收参数 b 的函数,并且这个函数又返回一个接收参数 c,
// 返回值为 Int 类型的函数) // 定义一个接收参数 a,并且返回一个接收参数 b 的函数,并且这个函数又返回一个接收参数 c,返回值为 Int 类型的函数。
func add(a: Int) -> (b: Int) -> (c: Int) -> Int { // 返回一个接收参数 b 的函数,并且这个函数又返回一个接收参数 c,返回值为 Int 类型的函数
return { (b:Int) -> (c: Int) -> Int in // 返回一个接收余下第一个参数 c,并且返回结果为 Int 类型的函数
return { (c: Int) -> Int in return a + b + c;
}
}
}
}
1.4 如何调用柯里化函数
创建柯里化类的实例
var curryInstance = Currying()
手动实现的柯里化函数调用
var result: Int = curryInstance.add(a: 10)(b: 20)(c: 30)
- 可能很多人都是第一次看这样的调用,感觉有点不可思议。
- 让我们回顾下 OC 创建对象
[[Person alloc] init],这种写法应该都见过吧,就是一下发送了两个消息,alloc返回一个实例,再用实例调用init初始化,上面也是一样,一下调用多个函数,每次调用都会返回一个函数,然后再次调用这个返回的函数。
手动实现的柯里化函数拆解调用
curryInstance.add(a: 10)调用一个接收参数a,并且返回一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数。// functionB: 一个接收参数 b 的函数,并且这个函数又返回一个接收参数 c,返回值为 Int 类型的函数
let functionB = curryInstance.add(a: 10)
functionB(b: 20)调用一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数。// functionC: 一个接收参数 c,返回值为 Int 类型的函数
let functionC = functionB(b: 20)
functionC(c: 30)调用一个接收参数c,返回值为Int类型的函数。// result: 函数的返回值
var result: Int = functionC(c: 30);
系统的柯里化函数调用
var result: Int = curryInstance.addCur(a: 10)(b: 20)(c: 30)
系统的柯里化函数拆解调用
curryInstance.addCur(a: 10)调用一个接收参数a,并且返回一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数。// Swift是强类型语言,这里没有报错,说明调用系统柯里化函数返回的类型和手动的 functionB 类型一致
// functionB: 一个接收参数 b 的函数,并且这个函数又返回一个接收参数 c,返回值为 Int 类型的函数
functionB = curryInstance.addCur(a: 10)
functionB(b: 20)调用一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数。// functionC: 一个接收参数c,返回值为Int类型的函数
functionC = functionB(b: 20)
functionC(c: 30)调用一个接收参数c,返回值为Int类型的函数。// result: 函数的返回值
result = functionC(c: 30) // 打印 60,60,60 说明手动实现的柯里化函数,和系统的一样。
print("\(r), \(res), \(result)")
1.5 柯里化函数使用注意
必须按照参数的定义顺序来调用柯里化函数,否则就会报错。
柯里化函数的函数体只会执行一次,只会在调用完最后一个参数的时候执行柯里化函数体。
以下调用
functionC(c: 30)才会执行函数体,这个可以自己断点调试。// curried: 柯里化函数
func addCur(a: Int)(b: Int)(c: Int) -> Int { println("\(a) + \(b) + \(c)")
return a + b + c
} // 创建柯里化类的实例
var curryInstance = Currying() // 不会执行柯里化函数体
functionB = curryInstance.addCur(a: 10) // 不会执行柯里化函数体
functionC = functionB(b: 20) // 执行柯里化函数体
result = functionC(c: 30)
1.6 柯里化函数的好处
这里就需要了解函数式编程思想了,柯里化函数就是运用了函数式编程思想,推荐看这篇文章函数式编程初探。
特点
- 1)只用 “表达式” (表达式: 单纯的运算过程,总是有返回值),不用 “语句” (语句: 执行某种操作,没有返回值)。
- 2)不修改值,只返回新值。
好处
- 1)代码简洁。
- 2)提高代码复用性。
- 3)代码管理方便,相互之间不依赖,每个函数都是一个独立的模块,很容易进行单元测试。
- 4)易于 “并发编程”,因为不修改变量的值,都是返回新值。
1.7 柯里化函数的实用性
实用性一:复用性
需求 1:地图类产品,很多界面都有相同的功能模块,比如搜索框。
- 我们可以利用柯里化函数,来组装界面,把界面分成一个个小模块,这样其他界面有相同的模块,直接运用模块代码,去重新组装下就好了。
实用性二:延迟性
柯里化函数代码需要前面的方法调用完成之后,才会来到柯里化函数代码中。
需求 2:阅读类产品,一个界面的显示,依赖于数据,需要加载完数据之后,才能判断界面显示。
- 这时候也可以利用柯里化函数,来组装界面,把各个模块加载数据的方法抽出来,等全部加载完成,再去执行柯里化函数,柯里化函数主要实现界面的组装。
举例说明
// 组合接口
// 为什么要定义接口,为了程序的扩展性,以后只需要在接口中添加对应的组合方法就好了。
protocol CombineUI { func combine(top: () -> ())(bottom: () -> ())()
} // 定义一个界面类,遵守组合接口
class UI: CombineUI { func combine(top: () -> ())(bottom: () -> ())() { // 搭建顶部
top() // 搭建底部
bottom()
}
}
2、Swift 中实例方法的柯里化
- Swift 中实例方法就是一个柯里化函数。
2.1 Swift 中实例方法的柯里化调用
Swift 中实例方法的柯里化调用
示例结构体
struct Example { var internalStr = "" func combine(externalStr: String) {
print(internalStr + " " + externalStr)
}
}
调用实例方法的常用格式
let example = Example(internalStr: "hello") example.combine(externalStr: "word") // hello word
调用实例方法的柯里化格式
let example = Example(internalStr: "hello") Example.combine(example)(externalStr: "word") // hello word
Swift 柯里化的更多相关文章
- Swift # 柯里化函数
前言 此次文章,讲述的是Swift的一个新特性(柯里化函数),可能很多iOS开发人员是第一次听这个词汇,包括我自己也是,自己也用了几天时间才总结出来,希望能帮助到各位咯,个人感觉偏向有开发经验的码友, ...
- swift 学习(二)基础知识 (函数,闭包,ARC,柯里化,反射)
函数 func x(a:Int, b:Int) {} func x(a:Int, b:Int) -> Void {} func x(a:Int, b:Int) ->(Int,Int ...
- Swift开发第四篇——柯里化
本篇分为两部分: 一.柯里化的基本使用 二.柯里化的使用场景 一.柯里化的基本使用 柯里化(Currying):也就是把接受多个参数的方法变换成接受第一个参数的方法,并且返回接受余下的参数并且返回结果 ...
- Swift函数柯里化(Currying)简谈
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 下面简单说说Swift语言中的函数柯里化.简单的说就是把接收多 ...
- Swift 4.0 废弃的柯里化
// 柯里化 // http://www.jianshu.com/p/6eaacadafa1a Swift 2.0 柯里化方法 (废弃) / ...
- JS 函数的柯里化与反柯里化
===================================== 函数的柯里化与反柯里化 ===================================== [这是一篇比较久之前的总 ...
- JS - 柯里化
一:what's this? 柯里化: 是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术.其实,柯里化就是用闭包原理实现函数 ...
- JAVAScript柯里化、部分应用参数终极理解
一.柯里化 在定义柯里化.部分应用参数的概念前,首先必须对闭包有深入的了解和定义,闭包一句话说清楚:函数返回值为函数. 柯里化的定义:将多参函数分解为按步骤接受单个参数的函数,如下代码: var mo ...
- 【转载】JS中bind方法与函数柯里化
原生bind方法 不同于jQuery中的bind方法只是简单的绑定事件函数,原生js中bind()方法略复杂,该方法上在ES5中被引入,大概就是IE9+等现代浏览器都支持了(有关ES5各项特性的支持情 ...
随机推荐
- Mybatis中的like模糊查询四种方式
1. 参数中直接加入%% param.setUsername("%CD%"); param.setPassword("%11%"); <select i ...
- 【前端基础系列】理解GET与POST请求区别
语义区别 GET请求用于获取数据 POST请求用于提交数据 缓存 GET请求能被缓存,以相同的URL再去GET请求会返回304 POST请求不能缓存 数据长度 HTTP协议从未规定过GET/POST请 ...
- dll反编译工具总结
有好多.net程序有加密狗或者有验证,如果exe或dll没有做过特殊处理,破解.net程序其实很简单,不过你要有足够的耐心! 我只做个简单的小例子,把公司的软件破解了,不要被老大知道,吼吼~~~~ 1 ...
- mysql explain优化
简介 MySQL 提供了一个 EXPLAIN 命令, 它可以对 SELECT 语句进行分析, 并输出 SELECT 执行的详细信息, 以供开发人员针对性优化.EXPLAIN 命令用法十分简单, 在 S ...
- 关于SSH不能连接及报错的问题总结
前言 此文不涉及到因网络.防火墙设备而导致的SSH不能访问.运维常见问题,这里不做过多的讲解,主要讲讲出了大家所知道的,还有其他什么原因会导致SSH无法访问呢?好了,那么,如果想知道的话,那就继续往下 ...
- Spring日记_01 之 Maven搭建 - 阿里云镜像替换maven配置文件
# 项目环境搭建 ## 1. Eclipse + Maven 中央库太慢所以使用 阿里云镜像:maven.aliyun.com 1. 配置Aliyun Maven 仓库 导入配置文件 set ...
- python开发环境PyCharm安转注册
0x1 ,安装 0x2 , 调整时间到2038年. 0x3 ,申请30天试用 0x4, 退出pycharm 0x5, 时间调整回来. ##注册方法2### 注册方法:在注册时选择 License se ...
- hdu 2036 求多边形面积 (凸、凹多边形)
<题目链接> Problem Description “ 改革春风吹满地,不会AC没关系;实在不行回老家,还有一亩三分地.谢谢!(乐队奏乐)” 话说部分学生心态极好,每天就知道游戏,这次考 ...
- windows server 2003 安全加固(二)
windows server 2003 安全加固 关闭默认端口 我们知道远程桌面服务端口默认开启在3389端口,如果我们一定要用到,最好能换到另外的端口上,放到靠后的端口号上去,比如10001. 更改 ...
- 模拟页面获取的php数据(三)
<?php return array( "aData" => [//通勤方式 "trafficType" => [ 0 => [ &qu ...