您的位置:首页 > 移动开发 > Swift

Swift中闭包的简单使用

2020-04-22 01:18 691 查看

闭包的定义

在Swift开发文档中是这样介绍闭包的:闭包是可以在你的代码中被传递和引用的功能性独立模块。Swift 中的闭包和 C 以及 Objective-C 中的 block 很像,还有其他语言中的匿名函数也类似。闭包的作用主要是:能够捕获和存储定义在其上下文中的任何常量和变量的引用, 能够为你处理所有关于捕获的内存管理的操作。

闭包的表达式语法

闭包表达式语法有如下的一般形式:

{ (parameters/接收的参数) -> (return type/闭包返回值类型) in
statements/保存在闭包中需要执行的代码
}

闭包根据你的需求是有类型的,闭包的类型 一般形式如下:

(parameters/接收的参数) -> (return type/闭包返回值类型)

利用typealias为闭包类型定义别名

这里先介绍一下 typealias的使用 : typealias是Swift中用来为已经存在的类型重新定义名字的关键字(类似于OC语法中的 typedef),重新命名的新名字用来替代之前的类型,并且能够使代码变得更加清晰简单容易理解。typealias 的用法很简单,直接用 = 赋值就可以了:

typealias <type name> = <type expression>

这里我们可以用 typealias 来为看似较为复杂的闭包类型定义别名,这样以后我们就可以用别名直接去申明这样类型的闭包了,例子如下:

//为没有参数也没有返回值的闭包类型起一个别名
typealias Nothing = () -> ()

//如果闭包的没有返回值,那么我们还可以这样写,
typealias Anything = () -> Void

//为接受一个Int类型的参数不返回任何值的闭包类型 定义一个别名:PrintNumber
typealias PrintNumber = (Int) -> ()

//为接受两个Int类型的参数并且返回一个Int类型的值的闭包类型 定义一个别名:Add
typealias Add = (Int, Int) -> (Int)

闭包是否接受参数、接受几个参数、返回什么类型的值完全取决于你的需求。

闭包的创建、赋值、调用

闭包表达式语法能够使用常量形式参数、变量形式参数和输入输出形式参数,但不能提供默认值。可变形式参数也能使用,但需要在形式参数列表的最后面使用。元组也可被用来作为形式参数和返回类型。在闭包的中会用到一个关键字in,in 可以看做是一个分割符,他把该闭包的类型和闭包的函数体分开,in前面是该闭包的类型,in后面是具体闭包调用时保存的需要执行的代码。表示该闭包的形式参数类型和返回类型定义已经完成,并且闭包的函数体即将开始执行。这里总结了一下可能用到的几种形式实现闭包的创建、赋值、调用的过程。例子如下:

方式一:利用typealias最完整的创建

//为(_ num1: Int, _ num2: Int) -> (Int) 类型的闭包定义别名:Add
typealias Add = (_ num1: Int, _ num2: Int) -> (Int)
//创建一个 Add 类型的闭包常量:addCloser1
let addCloser1: Add
//为已经创建好的常量 addCloser1 赋值
addCloser1 = {
(_ num1: Int, _ num2: Int) -> (Int) in
return num1 + num2
}
//调用闭包并接受返回值
let result = addCloser1(20, 10)

形式二:闭包类型申明和变量的创建合并在一起

//创建一个 (_ num1: Int, _ num2: Int) -> (Int) 类型的闭包常量:addCloser1
let addCloser1: (_ num1: Int, _ num2: Int) -> (Int)
//为已经创建好的常量 addCloser1 赋值
addCloser1 = {
(_ num1: Int, _ num2: Int) -> (Int) in
return num1 + num2
}
//调用闭包并接受返回值
let result = addCloser1(20, 10)

形式三:省略闭包接收的形参、省略闭包体中返回值

//创建一个 (Int, Int) -> (Int) 类型的闭包常量:addCloser1
let addCloser1: (Int, Int) -> (Int)
//为已经创建好的常量 addCloser1 赋值
addCloser1 = {
(num1, num2) in
return num1 + num2
}
//调用闭包并接受返回值
let result = addCloser1(20, 10)

形式四:在形式三的基础上进一步精简

//创建一个 (Int, Int) -> (Int) 类型的闭包常量:addCloser1 并赋值
let addCloser1: (Int, Int) -> (Int) = {
(num1, num2) in
return num1 + num2
}
//调用闭包并接受返回值
let result = addCloser1(20, 10)

形式五:如果闭包没有接收参数省略in

//创建一个 () -> (String) 类型的闭包常量:addCloser1 并赋值
let addCloser1: () -> (String) = {
return "这个闭包没有参数,但是有返回值"
}
//调用闭包并接受返回值
let result = addCloser1()

形式六:简写的实际参数名

//创建一个 (String, String) -> (String) 类型的闭包常量:addCloser1 并赋值
let addCloser1: (String, String) -> (String) = {
return "闭包的返回值是:\($0),\($1)"
}
//调用闭包并接受返回值
let result = addCloser1("Hello", "Swift!")

说明: 得益于Swift的类型推断机制,我们在使用闭包的时候可以省略很多东西,而且Swift自动对行内闭包提供简写实际参数名,你也可以通过 $0, $1, $2 这样的语法来引用闭包的实际参数值。如果你在闭包表达式中使用这些简写实际参数名,那么你可以在闭包的实际参数列表中忽略对其的定义,并且简写实际参数名的数字和类型将会从期望的函数类型中推断出来。in关键字也能被省略,$0 和 $1 分别是闭包的第一个和第二个 String类型的 实际参数(引自文档翻译)。

  • 点赞
  • 收藏
  • 分享
  • 文章举报
深爱JAVA 发布了5 篇原创文章 · 获赞 3 · 访问量 129 私信 关注
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: