1. * 闭包(Closures)
  2. * 闭包是自包含的功能代码块,可以在代码中使用或者用来作为参数传值。
  3. * 在Swift中的闭包与C、OC中的blocks和其它编程语言(如Python)中的lambdas类似。
  4. * 闭包可以捕获和存储上下文中定义的的任何常量和变量的引用。这就是所谓的变量和变量的自封闭,
  5. * 因此命名为”闭包“("Closures)").Swift还会处理所有捕获的引用的内存管理。
  6. *
  7. * 全局函数和嵌套函数其实就是特殊的闭包。
  8. * 闭包的形式有:
  9. * (1)全局函数都是闭包,有名字但不能捕获任何值。
  10. * (2)嵌套函数都是闭包,且有名字,也能捕获封闭函数内的值。
  11. * (3)闭包表达式都是无名闭包,使用轻量级语法,可以根据上下文环境捕获值。
  12. *
  13. * Swift中的闭包有很多优化的地方:
  14. * (1)根据上下文推断参数和返回值类型
  15. * (2)从单行表达式闭包中隐式返回(也就是闭包体只有一行代码,可以省略return)
  16. * (3)可以使用简化参数名,如$0, $1(从0开始,表示第i个参数...)
  17. * (4)提供了尾随闭包语法(Trailing closure syntax)
  18. */
  19. // 下面用Swift标准库中的sort方法来一步步简化闭包写法
  20. // sort函数需要两个参数
  21. // 参数一:数组
  22. // 参数二:一个闭包:带有两个参数,这两个参数类型与数组中的元素类型相同,返回值是Bool
  23. var names = ["Swift", "Arial", "Soga", "Donary"]
  24. // 第一种方式:使用函数
  25. func backwards(firstString: String, secondString: String) -> Bool {
  26. return firstString > secondString // 升序排序
  27. }
  28. // 这里第二个参数,传了一个函数
  29. // reversed is equal to ["Swift", "Soga", "Donary", "Arial"]
  30. var reversed = sort(nams, backwards)
  31. // 第二种方式:使用闭包方式
  32. // 完整闭包写法是在花括号内有参数列表和返回值,用关键字in表明闭包体的开始
  33. // (firstString: String, secondString: String) 闭包参数列表
  34. // -> Bool 指明闭包返回值类型是Bool
  35. // in关键字表明闭包体的开始
  36. reversed = sort(names, { (firstString: String, secondString: String) -> Bool in
  37. return firstString > secondString
  38. })
  39. // 这里可以进一步简化写法,因为闭包代码比较短,可以写到一行上
  40. reversed = sort(names, { (firstString: String, secondString: String) -> Bool in return firstString > secondString})
  41. // 下面再进一步简化写法 :根据环境上下文自动推断出类型
  42. // 参数列表都没有指明类型,也没有指明返回值类型,这是因为swift可以根据上下文推测出
  43. // firstString和secondString的类型会是names数组元素的类型,而返回值类型会根据return语句结果得到
  44. reversed = sort(names, { firstString, secondString in return firstString > secondString})
  45. // 再进一步简化:隐式返回(单行语句闭包)
  46. // 因为闭包体只有一行代码,可以省略return
  47. reversed = sort(names, { firstString, secondString in firstString > secondString})
  48. // 再进一步简化:使用简化参数名($i,i=0,1,2...从0开始的)
  49. // Swift会推断出闭包需要两个参数,类型与names数组元素相同
  50. reversed = sort(names, { $0 > $1 })
  51. // 最简单的一种写法:使用操作符
  52. reversed = sort(names, >)
  53. /*
  54. * 尾随闭包(Trailing Closures)
  55. * 如果函数需要一个闭包参数作为参数,且这个参数是最后一个参数,而这个闭包表达式又很长时,
  56. * 使用尾随闭包是很有用的。尾随闭包可以放在函数参数列表外,也就是括号外。如果函数只有一个参数,
  57. * 那么可以把括号()省略掉,后面直接跟着闭包。
  58. */
  59. // Array的方法map()就需要一个闭包作为参数
  60. let strings = numbers.map { // map函数后面的()可以省略掉
  61. (var number) -> String in
  62. var output = ""
  63. while number > 0 {
  64. output = String(number % 10) + output
  65. number /= 10
  66. }
  67. return output
  68. }
  69. /* 捕获值
  70. * 闭包可以根据环境上下文捕获到定义的常量和变量。闭包可以引用和修改这些捕获到的常量和变量,
  71. * 就算在原来的范围内定义为常量或者变量已经不再存在(很牛逼)。
  72. * 在Swift中闭包的最简单形式是嵌套函数。
  73. */
  74. func increment(#amount: Int) -> (() -> Int) {
  75. var total = 0
  76. func incrementAmount() -> Int {
  77. total += amount // total是外部函数体内的变量,这里是可以捕获到的
  78. return total
  79. }
  80. return incrementAmount // 返回的是一个嵌套函数(闭包)
  81. }
  82. // 闭包是引用类型,所以incrementByTen声明为常量也可以修改total
  83. let incrementByTen = increment(amount: 10)
  84. incrementByTen() // return 10,incrementByTen是一个闭包
  85. // 这里是没有改变对increment的引用,所以会保存之前的值
  86. incrementByTen() // return 20
  87. incrementByTen() // return 30
  88. let incrementByOne = increment(amount: 1)
  89. incrementByOne() // return 1
  90. incrementByOne() // return 2
  91. incrementByTen() // return 40
  92. incrementByOne() // return 3

Swift学习之十四:闭包(Closures)的更多相关文章

  1. Swift学习笔记十四:构造(Initialization)

         类和结构体在实例创建时,必须为全部存储型属性设置合适的初始值. 存储型属性的值不能处于一个未知的状态.     你能够在构造器中为存储型属性赋初值,也能够在定义属性时为其设置默认值.下面章节 ...

  2. Swift学习笔记十四

    Deinitialization 当类的实例对象即将要被释放时,会立即调用deinitializer,通过deinit关键字来定义deinitializer,和initializer一样,它也只存在于 ...

  3. swift学习第十四天:属性监听器

    监听属性的改变 在OC中我们可以重写set方法来监听属性的改变 Swift中可以通过属性观察者来监听和响应属性值的变化 通常是监听存储属性和类属性的改变.(对于计算属性,我们不需要定义属性观察者,因为 ...

  4. python3.4学习笔记(十四) 网络爬虫实例代码,抓取新浪爱彩双色球开奖数据实例

    python3.4学习笔记(十四) 网络爬虫实例代码,抓取新浪爱彩双色球开奖数据实例 新浪爱彩双色球开奖数据URL:http://zst.aicai.com/ssq/openInfo/ 最终输出结果格 ...

  5. Linux学习之十四、管线命令

    Linux学习之十四.管线命令 地址:http://vbird.dic.ksu.edu.tw/linux_basic/0320bash_6.php

  6. 风炫安全WEB安全学习第二十四节课 利用XSS钓鱼攻击

    风炫安全WEB安全学习第二十四节课 利用XSS钓鱼攻击 XSS钓鱼攻击 HTTP Basic Authentication认证 大家在登录网站的时候,大部分时候是通过一个表单提交登录信息. 但是有时候 ...

  7. (C/C++学习笔记) 十四. 动态分配

    十四. 动态分配 ● C语言实现动态数组 C语言实现动态数组,克服静态数组大小固定的缺陷 C语言中,数组长度必须在创建数组时指定,并且只能是一个常数,不能是变量.一旦定义了一个数组,系统将为它分配一个 ...

  8. swift学习第十五天:闭包

    闭包 闭包的介绍 闭包和OC中的block非常相似 OC中的block是匿名的函数 Swift中的闭包是一个特殊的函数 block和闭包都经常用于回调 注意:闭包和block一样,第一次使用时可能不习 ...

  9. Swift 学习笔记(四)

    116.使用可选链式调用代替强制展开 通过在想调用的属性.方法.或下标的可选值(optional value)后面放一个问号(?),可以定义一个可选链.这一点很像在可选值后面放一个叹号(!)来强制展开 ...

随机推荐

  1. 关于LWIP---UDP

    lwip是一个轻量级的TCP/IP协议栈(Lightweight TCP/IP Stack)实现,最初是瑞士计算机科学学院Adam Dunkels编写的一个应用于无操作系统的嵌入式系统中的TCP/IP ...

  2. WPS Office手机版调用接口代码指导帖之一(Android)

    经常会有一些喜欢开发鼓捣的童鞋问我们,WPS Office手机版是否提供调用接口,希望在android中使用一个调用命令,直接调用WPS手机版来打开指定的DOC文件,而不用弹出一个程序可选列表(如果用 ...

  3. javascript函数值的重写

    原文:javascript函数值的重写 javascript函数值的重写 定义了一个函数,需要重写这个函数并使用原先的函数值.做法是: 1.定义一个变量让原先函数的值指向它,把原先函数的指向一个新的函 ...

  4. 在一个frame设置四个组件

    import javax.swing.*; import java.awt.event.*; import java.awt.*; class TouChaCol{ JFrame frame; JLa ...

  5. python写xml文件

    为了便于后续的读取处理,这里就将信息保存在xml文件中,想到得到的文件如下: 1 <?xml version="1.0" encoding="utf-8" ...

  6. ios 后台模式

    1.在后台可以继续播放音频 To play sound in the background, make sure to add the following to the Info.plist file ...

  7. Xcode7.3.1真机调试ios10

    如果自己的ios测试机不小心升级到比Xcode更高的ios系统, 那么这时候是无法使用真机来进行调试的. 但是我们可以通过拷贝与测试机一样版本的系统来解决这个问题. 去下载一个Xcode8,然后安装, ...

  8. 利用协议代理实现导航控制器UINavigationController视图之间的正向传值和反向传值

    实验说明 (1)正向传值:比如A类里地值要传给B类用,就是我们先在A类中声明一个B类对象(当然B类头文件要import过来),然后把A类中得某个 值传递给B类中得某个值(所以需要在B类中先准备一个变量 ...

  9. swift 用协议实现代理传值功能

    1.功能简介 RootViewController中用个lable和一个按钮,点击按钮跳转到模态窗口.在模态窗口中有个TextField和一个按钮,输入文字点击关闭模态按钮后跳转到RootViewCo ...

  10. navigationController显示隐藏问题

    今天遇到设置: self.navigationController.navigationBarHidden= YES; 点击返回上一个UIViewController的时候这个时候这个navigati ...