Swift闭包概念与常见使用场景总结
·Swift 闭包
闭包(Closures)是自包含的功能代码块,可以在代码中使用或者用来作为参数传值。
Swift 中的闭包与 C 和 Objective-C 中的代码块(blocks)以及其他一些编程语言中的 匿名函数比较相似。
全局函数和嵌套函数其实就是特殊的闭包。
闭包的形式有:
|
全局函数 |
嵌套函数 |
闭包表达式 |
|
有名字但不能捕获任何值。 |
有名字,也能捕获封闭函数内的值。 |
无名闭包,使用轻量级语法,可以根据上下文环境捕获值。 |
Swift中的闭包有很多优化的地方:
- 根据上下文推断参数和返回值类型
- 从单行表达式闭包中隐式返回(也就是闭包体只有一行代码,可以省略return)
- 可以使用简化参数名,如$0, $1(从0开始,表示第i个参数...)
- 提供了尾随闭包语法(Trailing closure syntax)
闭包表达式
闭包表达式是一种利用简洁语法构建内联闭包的方式。
闭包表达式提供了一些语法优化,使得撰写闭包变得简单明了。
sort 函数
Swift 标准库提供了名为sort的函数,会根据您提供的用于排序的闭包函数将已知类型数组中的值进行排序。
排序完成后,sort(_:)方法会返回一个与原数组大小相同,包含同类型元素且元素已正确排序的新数组,原数组不会被sort(_:)方法修改。
sort(_:)方法需要传入两个参数:
- 已知类型的数组
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闭包概念与常见使用场景总结的更多相关文章
- JS闭包的理解及常见应用场景
JS闭包的理解及常见应用场景 一.总结 一句话总结: 闭包是指有权访问另一个函数作用域中的变量的函数 1.如何从外部读取函数内部的变量,为什么? 闭包:f2可以读取f1中的变量,只要把f2作为返回值, ...
- Swift --闭包表达式与闭包(汇编分析)
在Swift中,可以通过func定义一个函数,也可以通过闭包表达式定义一个函数! 一.闭包表达式 概念 闭包表达式与定义函数的语法相对比,有区别如下: 去除了func 去除函数名 返回值类型添加了关键 ...
- Swift闭包
把Swift中的 block 常见的声明和写法作一个总结.以免后续忘了,好查阅. // // blockDemo.swift // swiftDemo // // Created by appl ...
- javascript,jquery(闭包概念)(转)
偶尔听人说javascript闭包,让我联想起以前学编译原理和数字逻辑里讲的闭包,以前上课讲的闭包很难懂,而且含有递归的意思在里面,现在不想再查看里面的闭包概念. 但javascript我是经常要用, ...
- Git 基础 —— 常见使用场景
Git 基础学习系列 Git 基础 -- 安装 配置 别名 对象 Git 基础 -- 常用命令 Git 基础 -- 常见使用场景 Git基础 -- Github 的使用 突然插入 Bugifx 工作, ...
- axios基于常见业务场景的二次封装
axios axios 是一个基于 promise 的 HTTP 库,可以用在浏览器和 node.js 中.在前端框架中的应用也是特别广泛,不管是vue还是react,都有很多项目用axios作为网络 ...
- 浅析阿里云API网关的产品架构和常见应用场景
自上世纪60年代计算机网络发展开始,API(Application Programming Interface )随之诞生,API即应用程序接口,是实现系统间衔接的桥梁.时至今日,API市场已经形成了 ...
- ACCESS常见错误场景
ACCESS常见错误场景 今天用access时发现好多报错的地方,emmm,比MySQL麻烦好多,有些甚至还要自己去配置环境 不吐槽了,进入正题: 报错场景一:您尝试执行不包含指定聚合函数的查询 第一 ...
- postman中环境变量的设置方法、使用方法和实际中常见使用场景
文中共介绍2种添加环境变量的方法.2种使用环境变量的方法,以及不同方法的适用范围. 文中给出了环境变量的两种常见使用场景:切换环境.动态参数关联(前一个请求的响应作为下一个请求的入参) 2种添加环境变 ...
随机推荐
- ajax返回值给上层函数的方法。
function load_val(callback){//定义一个回调函数 $.getJSON('test.php' , function(dat){ callback(data);//将返回结果当 ...
- shell字符串判空
主要用到两个命令 -n -z if [ -n "$PID" ]; then echo "PID is not empty" fi if[ -z "$ ...
- 解析利用wsdl.exe生成webservice代理类的详解
利用wsdl.exe生成webservice代理类:根据提供的wsdl生成webservice代理类1.开始->程序->Visual Studio 2005 命令提示2.输入如下红色标记部 ...
- js中RegExp类型
ECMAScript通过RegExp类型来支持正则表达式. var expression = / pattern / flag ; pattern可以是任意的正则表达式.每个正则都带有标志,用以正则表 ...
- linux 服务器登录显示lastlogin
1.参数修改: /etc/ssh/sshd_config 问价里边的 printlastlog 设置为yes /etc/ssh/sshd_config 问价里边的 printmotd 设置为yes 2 ...
- I/O阻塞非阻塞,同步异步
http://www.cnblogs.com/luotianshuai/p/5098408.html "阻塞"与"非阻塞"与"同步"与&qu ...
- mysql 更改自动增长字段值的重新设定
今天在服务器上MYSQL库里的一个表插入数据,主键id是auto_increment自动增长类型的.发现插入的值从2247734开始,而实际上id的最大值才22722,不明原因. 删除了新增的,opt ...
- 一个简单的SNTP客户端
借鉴于python网络编程攻略 #/usr/local/bin/python3.5 #coding:utf-8 import socket, struct, time NTP_server = &qu ...
- Spark Streaming
Spark Streaming Spark Streaming 是Spark为了用户实现流式计算的模型. 数据源包括Kafka,Flume,HDFS等. DStream 离散化流(discretize ...
- 如何在asp.net中使用多线程及队列,异步处理一个耗时的任务(原创)
最近想在使用.net 的队列处理一些耗时的工作.经过考虑,需要先设计一个类,类中包含一个静态的队列.主要是写队列和读取队列. public class PaperCalculator { // 用于存 ...