·Swift 闭包

闭包(Closures)是自包含的功能代码块,可以在代码中使用或者用来作为参数传值。

Swift 中的闭包与 C 和 Objective-C 中的代码块(blocks)以及其他一些编程语言中的 匿名函数比较相似。

全局函数和嵌套函数其实就是特殊的闭包。

闭包的形式有:

全局函数

嵌套函数

闭包表达式

有名字但不能捕获任何值。

有名字,也能捕获封闭函数内的值。

无名闭包,使用轻量级语法,可以根据上下文环境捕获值。

Swift中的闭包有很多优化的地方:

  1. 根据上下文推断参数和返回值类型
  2. 从单行表达式闭包中隐式返回(也就是闭包体只有一行代码,可以省略return)
  3. 可以使用简化参数名,如$0, $1(从0开始,表示第i个参数...)
  4. 提供了尾随闭包语法(Trailing closure syntax)

闭包表达式

闭包表达式是一种利用简洁语法构建内联闭包的方式。
闭包表达式提供了一些语法优化,使得撰写闭包变得简单明了。

sort 函数
Swift 标准库提供了名为sort的函数,会根据您提供的用于排序的闭包函数将已知类型数组中的值进行排序。

排序完成后,sort(_:)方法会返回一个与原数组大小相同,包含同类型元素且元素已正确排序的新数组,原数组不会被sort(_:)方法修改。

sort(_:)方法需要传入两个参数:

  1. 已知类型的数组
    2.   闭包函数,该闭包函数需要传入与数组元素类型相同的两个值,并返回一个布尔类型值来表明当排序结束后传入的第一个参数排在第二个参数前面还是后面。如果第一个参数值出现在第二个参数值前面,排序闭包函数需要返回true,反之返回false。

参数名称缩写
Swift 自动为内联函数提供了参数名称缩写功能,您可以直接通过$0,$1,$2来顺序调用闭包的参数。

import Cocoa

let names = ["AT",
"AE", "D", "S", "BE"]

var reversed =
names.sort( { $0 > $1 } )

print(reversed)

$0和$1表示闭包中第一个和第二个String类型的参数。

以上程序执行输出结果为:

["S", "D", "BE", "AT", "AE"]

运算符函数

实际上还有一种更简短的方式来撰写上面例子中的闭包表达式。

Swift 的String类型定义了关于大于号 (>) 的字符串实现,其作为一个函数接受两个String类型的参数并返回Bool类型的值。 而这正好与sort(_:)方法的第二个参数需要的函数类型相符合。 因此,您可以简单地传递一个大于号,Swift可以自动推断出您想使用大于号的字符串函数实现:

import Cocoa

let names = ["AT", "AE", "D", "S", "BE"]

var reversed = names.sort(>)

print(reversed)

以上程序执行输出结果为:

["S", "D", "BE", "AT", "AE"]

尾随闭包

尾随闭包是一个书写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用。

注意:如果函数只需要闭包表达式一个参数,当您使用尾随闭包时,您甚至可以把()省略掉。

func someFunctionThatTakesAClosure(closure: () -> Void) {
    // 函数体部分
}
 
// 以下是不使用尾随闭包进行函数调用
someFunctionThatTakesAClosure({
    // 闭包主体部分
})
 
// 以下是使用尾随闭包进行函数调用
someFunctionThatTakesAClosure() {
  // 闭包主体部分
}

捕获值

闭包可以在其定义的上下文中捕获常量或变量。

即使定义这些常量和变量的原域已经不存在,闭包仍然可以在闭包函数体内引用和修改这些值。

Swift最简单的闭包形式是嵌套函数,也就是定义在其他函数的函数体内的函数。

嵌套函数可以捕获其外部函数所有的参数以及定义的常量和变量。

闭包是引用类型。

总结:闭包的常见使用场景

目前基于Swift 3.0+ 版本总结闭包的常见使用场景

// 1.作为变量(类中的属性):

// var 闭包名称: (参数列表) -> 返回类型

var
closureName1: (_ name: String, _ age: Int) -> String

// 2.作为可选的变量(类中的属性):

// var
closureName: ((parameterTypes) -> returnType)?

var
closureName2: ((_ name: String, _ age: Int) -> String)?

// 3.作为类型别名(闭包类型):

// typealias
closureType = (parameterTypes) -> returnType

typealias
closureType = (_ name: String, _ age: Int) -> String

// 4.作为常量(类中的属性):

// let
closureName: closureType = 闭包表达式

let
closureName3: closureType = { (_ name: String, _ age: Int) -> String in

return "My name is \(name), age is \(age)"

}

closureName3("Abnerzj",
10)

// 5.定义函数时作为函数的参数:

// 5.1 func 函数名(参数名: 闭包类型)

func closureFuncName(closureParameterName:
closureType) -> Void {

// 函数体部分

}

// 5.2 func 函数名(参数名: 闭包表达式)

func closureFuncName2(closureParameterName:
(_ name: String, _ age: Int) -> String) -> Void {

// 函数体部分

}

// 6.调用函数时作为函数的参数:完整的闭包格式

// 函数名(参数名: 闭包表达式)

// 函数名(参数名: { 闭包参数列表 -> 闭包返回值 in 闭包函数体 })

closureFuncName(closureParameterName:
{

(_ name: String, _ age: Int) -> String in

return "My name is" + name + ",
age is \(age)"

})

// 7.调用函数时作为函数的参数:根据上下文推断类型:

// 函数名(参数名: { 实参名1, 实参名2 in 闭包函数体 })

closureFuncName(closureParameterName:
{

name, age in

return "My name is" + name + ",
age is \(age)"

})

// 8.调用函数时作为函数的参数:单行表达式闭包隐式返回,可以隐藏return关键字

// 函数名(参数名: { 实参名1, 实参名2 in 闭包函数体 })

closureFuncName(closureParameterName:
{

name, age in "My name is" + name
+ ", age is \(age)"

})

// 9.调用函数时作为函数的参数:参数名称缩写($0,$1,$2...来顺序代替参数列表中的参数名)

// 函数名(参数名: { 闭包函数体 })

closureFuncName(closureParameterName:
{

"My name is \($0), age is \($1)"

})

// 10.调用函数时作为函数的参数:尾随闭包(作为函数的最后一个参数),不是函数的唯一一个参数时

// 函数名() { 闭包函数体 }

closureFuncName()
{

"My name is \($0), age is \($1)"

}

// 11.调用函数时作为函数的参数:尾随闭包(作为函数的最后一个参数),是函数的唯一一个参数时

// 函数名 { 闭包函数体 }

closureFuncName
{

"My name is \($0), age is \($1)"

}

// 12.调用函数时传入一个闭包函数作为函数的参数

// 函数名(参数名: 闭包函数名)

func closureFunc(_
name: String, _ age: Int) -> String {

return name + "\(age)"

}

closureFuncName(closureParameterName:
closureFunc)

// 13.调用函数时作为函数的参数:循环强引用

// 函数名(参数名: { [弱引用或无主引用列表] 闭包参数列表 -> 闭包返回值 in 闭包函数体 })

// 第一种:推荐

closureFuncName(closureParameterName:
{

[weak self] (_ name: String, _ age: Int)
-> String in

self?.view.backgroundColor = UIColor.red

return "My name is" + name + ",
age is \(age)"

})

// 第二种:

weak var
weakself = self

closureFuncName(closureParameterName:
{

(_ name: String, _ age: Int) -> String in

weakself?.view.backgroundColor = UIColor.red

return "My name is" + name + ",
age is \(age)"

})

// 第三种:

closureFuncName(closureParameterName:
{

[unower self] (_ name: String, _ age: Int)
-> String in

self?.view.backgroundColor = UIColor.red

return "My name is" + name + ",
age is \(age)"

})

上面是我对Swift闭包的理解与总结。欢迎各位博友学习浏览,如在哪里总结的不到位希望各位朋友提出更好的建议。

此博文只是为了与更多博友们交流学习心得,如需转载请注明出处。 谢谢!

关于闭包的总结就到此处啦! 觉得此博文整理的好的话,就给个赞吧。。。 感谢大家的支持!!!

Swift闭包概念与常见使用场景总结的更多相关文章

  1. JS闭包的理解及常见应用场景

    JS闭包的理解及常见应用场景 一.总结 一句话总结: 闭包是指有权访问另一个函数作用域中的变量的函数 1.如何从外部读取函数内部的变量,为什么? 闭包:f2可以读取f1中的变量,只要把f2作为返回值, ...

  2. Swift --闭包表达式与闭包(汇编分析)

    在Swift中,可以通过func定义一个函数,也可以通过闭包表达式定义一个函数! 一.闭包表达式 概念 闭包表达式与定义函数的语法相对比,有区别如下: 去除了func 去除函数名 返回值类型添加了关键 ...

  3. Swift闭包

    把Swift中的 block 常见的声明和写法作一个总结.以免后续忘了,好查阅. // //  blockDemo.swift //  swiftDemo // //  Created by appl ...

  4. javascript,jquery(闭包概念)(转)

    偶尔听人说javascript闭包,让我联想起以前学编译原理和数字逻辑里讲的闭包,以前上课讲的闭包很难懂,而且含有递归的意思在里面,现在不想再查看里面的闭包概念. 但javascript我是经常要用, ...

  5. Git 基础 —— 常见使用场景

    Git 基础学习系列 Git 基础 -- 安装 配置 别名 对象 Git 基础 -- 常用命令 Git 基础 -- 常见使用场景 Git基础 -- Github 的使用 突然插入 Bugifx 工作, ...

  6. axios基于常见业务场景的二次封装

    axios axios 是一个基于 promise 的 HTTP 库,可以用在浏览器和 node.js 中.在前端框架中的应用也是特别广泛,不管是vue还是react,都有很多项目用axios作为网络 ...

  7. 浅析阿里云API网关的产品架构和常见应用场景

    自上世纪60年代计算机网络发展开始,API(Application Programming Interface )随之诞生,API即应用程序接口,是实现系统间衔接的桥梁.时至今日,API市场已经形成了 ...

  8. ACCESS常见错误场景

    ACCESS常见错误场景 今天用access时发现好多报错的地方,emmm,比MySQL麻烦好多,有些甚至还要自己去配置环境 不吐槽了,进入正题: 报错场景一:您尝试执行不包含指定聚合函数的查询 第一 ...

  9. postman中环境变量的设置方法、使用方法和实际中常见使用场景

    文中共介绍2种添加环境变量的方法.2种使用环境变量的方法,以及不同方法的适用范围. 文中给出了环境变量的两种常见使用场景:切换环境.动态参数关联(前一个请求的响应作为下一个请求的入参) 2种添加环境变 ...

随机推荐

  1. margin:0 auto在IE中失效的解决方案

    转自:http://www.cnblogs.com/hongchenok/archive/2012/11/29/2795041.html 最近在开发项目的时候,发现在火狐浏览器中设置外容器margin ...

  2. NHibernate系列文章八:NHibernate对象一级缓存

    摘要 Nhibernatea缓存非常强大,按照缓存存储在Session对象还是SessionFactory对象分为一级缓存和二级缓存. 一级缓存存在于Session对象里,也叫Session缓存,由S ...

  3. NHibernate系列文章十八:NHibernate关系之一对多(附程序下载)

    摘要 这篇文章介绍NHibernate最实用的内容:关系映射. NHibernate的关系映射方式有三种: Set:无序对象集合,集合中每一个元素不能重复. List:有序对象集合,集合中的元素可以重 ...

  4. 正则表达式python和C++对比

    pattern格式(基本通用): pattern格式 符号 说明 ^ 匹配开头 $ 匹配结尾 . 匹配任意一个字符 [...] 匹配任意一个指定的字符 [^...] 匹配任意一个非指定的字符 * 匹配 ...

  5. 在Egret实现二维码长按识别

      Egret中二维码图片,是在canvas上,无法在微信上长按扫描识别. 由于微信长按识别二维码是截屏扫描原理,所以只要长按当前屏幕任意一张图片,都能够识别当前屏幕上的二维码. 这里把二维码放在ex ...

  6. 使用WCF 测试客户端测试你的WCF服务

    wcftestclient.exe是一个GUI的工具用于测试WCF,只需在Visual studio command line 窗口中键入 wcftestclient,就启动这个程序.如下图: 然后通 ...

  7. 数据库dump导入

    数据库dump导入 一.导入命令介绍: Oracle dump数据导入导出有两种方式:imp/exp.impdp/expdp.两者区别: 1.exp/imp客户端程序,受网络,磁盘的影响:impdp/ ...

  8. PHP的轻量消息队列php-resque使用说明

    日志未经声明,均为AlloVince原创.版权采用『 知识共享署名-非商业性使用 2.5 许可协议』进行许可. 消息队列处理后台任务带来的问题 项目中经常会有后台运行任务的需求,比如发送邮件时,因为要 ...

  9. socket编程相关的结构体和字节序转换、IP、PORT转换函数

    注意:结构体之间不能直接进行强制转换, 必须先转换成指针类型才可以进行结构体间的类型转换, 这里需要明确的定义就是什么才叫强制转换. 强制转换是将内存中一段代码以另一种不同类型的方式进行解读, 因此转 ...

  10. Echarts x轴显示不全

    xAxis : [ { type : 'category', data : ['采矿业','制造业','电力热力燃气及水生产和供应业','建筑业'], axisTick: { alignWithLab ...