Swift的闭包(一):闭包简介、闭包表达式的优化
定义:Closures are self-contained blocks of functionality that can be passed around and used in your code.
跟oc中的block相似。
Capture can capture and store reference to any constants and variables from the context in which they are defined. This is known as closing over those constants and variables.
闭包以下面三种形式中的一种存在:
- 全局函数是一个有名字并且不捕获任何变量的闭包;
- 嵌套函数(nested function)是有一个名字并且可以从他们的外层函数(enclosing function)捕获变量的闭包;
- 闭包表达式是没有名字,可以从他们的surrounding context中捕获变量;
Swift的闭包有干净、清晰的格式,并且有足够的优化,包括:
- 从上下文中推断参数和返回值;
- Implicit returns from single-expression closures;
- 简短的参数名字;
- 尾闭包语法
闭包表达式
闭包表达式提供几种语法优化,下面的例子展示通过迭代提炼一个单一的表达式sorted(by:),来展示这些优化。
排序方法
Swift的标准库提供一个叫做sorted(by:)的方法,基于用户提供的一个排序闭包的输出,可以对一个已知类型的数组进行排序。排序完成后,这个方法返回一个新的数组(跟原来的数组有相同的类型和元素数量)。
下面闭包表达式的例子,使用sorted(by:)方法对一组String值进行逆字母序排序。初始的数组如下:
var names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
sorted(by:)方法接受一个带两个相同类型的参数并返回一个Bool值的闭包。如果第一个值需要出现在第二个值后边,闭包需要返回true,否则,返回false。
这个例子是对一组String值进行排序,所以排序的闭包的需要是一个(String, String)->Bool类型的函数。
下面用一个正常的函数来作为这个排序闭包:
func backward(_ s1: String, _ s2: String) -> Bool {
return s1 > s2
}
var reversedNames = names.sorted(by: backward);
闭包表达式的语法

闭包表达式里的patameters可以是in-out参数,但不能有默认值。
下面这段代码展示闭包表示式的版本:
reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in
return s1 > s2
})
从上下文推断类型
因为上面例子中的闭包作为一个参数传递给了一个方法,Swift可以推断它参数的类型和返回值的类型。这个sorted(by:)方法是在一个String数组上调用,因此它的参数必须是一个(String, String) -> Bool类型的函数。这意味着(String, String)和Bool类型不需要写出来作为闭包定义的一部分。因为所有的类型都可以推断出来,返回的箭头->和参数两边的小括号也可以被删除:
reversedNames = names.sorted(by: {s1, s2 in return s1 > s2})
从单表达式闭包隐式返回(Implicit Returns fro Single-Expression Closures)
单表达式闭包可以通过删除return关键字来隐式返回这个单表达式的结果:
reversedNames = names.sorted(by: {s1, s2 in s1 > s2})
简短的参数名字(Shorthand Argument Names)
Swift自动给内联闭包提供简短的参数名字,通过$0,$1,$2等来引用闭包的参数。
如果你在闭包雕大师中使用简短的参数名字,你可以从闭包的定义中删除闭包的参数列表,并且简短参数名字的数量和类型将通过预期的函数类型来推断。因为这个闭包表达式完全的组成了它的body,这个in关键字也可以删除。
reversedNames = names.sorted(by: {$ > $})
运算符函数(Operator Functions)
实际上,对于上面的例子有一个更简短的写法。Swift的String类型实现了针对String类型的大于操作符作为一个有两个String类型参数和Bool返回值的函数。恰好跟sorted(by:)的需求吻合。因此,你可以简单的传递一个大于操作符,Swift将推断你想使用针对String的实现:
reversedNames = names.sorted(by: >)
尾闭包(Trailing Closures)
如果你需要给一个函数传递一个闭包作为函数的最后的参数,并且闭包表达式很长,写成尾闭包是更好的选择。一个尾闭包被写在函数调用小括号的后边。当你写尾闭包语法的时候,你不用为函数调用的闭包写参数标签。
func someFunctionThatTasksAClosure(closure: () -> Void) {
}
//不使用尾闭包的函数调用
someFunctionThatTasksAClosure(closure: {
//closure's body
})
//使用尾闭包的函数调用
someFunctionThatTasksAClosure() {
//closure's body
}
排序的例子使用尾闭包语法可以这么写:
let reversedNames = names.sorted() {
(s1: String, s2: String) -> Bool in
return s1 > s2
}
或者,这样:
reversedNames = names.sorted() {$ > $}
如果闭包表达式是方法的唯一参数,并且你使用表达式作为尾闭包,那么在调用函数没有必要写小括号在函数名后边:
reversedNames = names.sorted {$ > $}
参考:The Swift Programming Language(Swift 3)
Swift的闭包(一):闭包简介、闭包表达式的优化的更多相关文章
- Swift学习之十四:闭包(Closures)
* 闭包(Closures) * 闭包是自包含的功能代码块,可以在代码中使用或者用来作为参数传值. * 在Swift中的闭包与C.OC中的blocks和其它编程语言(如Python)中的lambdas ...
- Swift学习(三):闭包(Closures)
定义 闭包(Closures)是独立的函数代码块,能在代码中传递及使用. 语法 {(parameters) -> return type in statements } 注:闭包表达式语法可以使 ...
- Swift语法基础入门三(函数, 闭包)
Swift语法基础入门三(函数, 闭包) 函数: 函数是用来完成特定任务的独立的代码块.你给一个函数起一个合适的名字,用来标识函数做什么,并且当函数需要执行的时候,这个名字会被用于“调用”函数 格式: ...
- Swift中文教程(四)--函数与闭包
原文:Swift中文教程(四)--函数与闭包 Function 函数 Swift使用func关键字来声明变量,函数通过函数名加小括号内的参数列表来调用.使用->来区分参数名和返回值的类型: fu ...
- 深入学习javaScript闭包(闭包的原理,闭包的作用,闭包与内存管理)
前言 虽然JavaScript是一门完整的面向对象的编程语言,但这门语言同时也拥有许多函数式语言的特性. 函数式语言的鼻祖是LISP,JavaScript在设计之初参考了LISP两大方言之一的Sche ...
- js闭包的作用域以及闭包案列的介绍:
转载▼ 标签: it js闭包的作用域以及闭包案列的介绍: 首先我们根据前面的介绍来分析js闭包有什么作用,他会给我们编程带来什么好处? 闭包是为了更方便我们在处理js函数的时候会遇到以下的几 ...
- print(函数.__closure__) 来判断是不是闭包, 返回cell , 是闭包, 返回None 则不是闭包
print(函数.__closure__) 来判断是不是闭包, 返回cell , 是闭包, None 则不是闭包
- Swift高阶函数介绍(闭包、Map、Filter、Reduce)
Swift语言有非常多函数式编程的特性.常见的map,reduce,filter都有,初看和python几乎相同,以下简介下 闭包介绍: 闭包是自包括的功能代码块,能够在代码中使用或者用来作为參数传值 ...
- 初探swift语言的学习笔记三(闭包-匿名函数)
作者:fengsh998 原文地址:http://blog.csdn.net/fengsh998/article/details/29353019 转载请注明出处 假设认为文章对你有所帮助,请通过留言 ...
随机推荐
- JQuery zoom插件学习
jquery zoom是一款图片放大插件,经常用在商城商品页面里. 使用JQuery zoom插件,除了需要引入JQuery.js外,还要引入JQuery.zoom.js文件及jqzoom.css文件 ...
- POJ3274-牛的属性-HASH-ACM
原题:POJ3274 参考:进击的阿俊 已知有n头牛,用一个K位二进制数Ak,Ak-1,...,A1表示一头牛具有的特征,Ai=1表示具有特征i.现给定按顺序排列的N头牛的k位特征值,称某个连续范围内 ...
- php获取https下的内容
直接用file_get_contents,会报错: $url = ('https://xxx.com"); file_get_contents($url); 错误: Warning: fil ...
- SharePoint 获取Lookup 字段的值
获取某个List里的Lookup字段的值是很普遍的事,那么我们将它封装起来 获取Lookup字段值的方法: /// <summary> /// To get lookup field Id ...
- cf D. Physical Education and Buns
http://codeforces.com/contest/394/problem/D 题意:给你n个数,然后通过操作使得这n个数变为一个等差数列,操作是可以经过小于等于k次加1或减去1,要使得k尽量 ...
- 利用js制作html table分页示例(js实现分页)
有时候table的列数太长,不利于使用者查询,所以利用JS做了一个table的分页,以下为相关代码 一.JS代码 <script type="text/javascript" ...
- c++ smart pointer
智能指针(smart pointer)是存储指向动态分配(堆)对象指针的类,用于生存期控制,能够确保自动正确的销毁动态分配的对象,防止内存泄露.它的一种通用实现技术是使用引用计数(reference ...
- [转]Android DPAD not enabled in AVD
转自:http://blog.csdn.net/flyhigh200703/article/details/8955484 问题描述:打开Android的仿真器,右侧的按键部分对于上下左右键出现以下 ...
- 【JavaScript】
右键禁用.防止文字选中 .返回选中的文本 JavaScript 原理 Javascript高性能动画与页面渲染 前端不为人知的一面--前端冷知识集锦 屏幕外去计算值,position:absolute ...
- Android常用的一些make命令(转载)--不错
原文网址:http://blog.sina.com.cn/s/blog_abc7e49a01011y0n.html 1.make -jXX XX表示数字,这个命令将编译Android系统并生成镜像, ...