- 921. Minimum Add to Make Parentheses Valid 使括号有效的最少添加
- 915. Partition Array into Disjoint Intervals 分割数组
- 932. Beautiful Array 漂亮数组
- 940. Distinct Subsequences II 不同的子序列 II
闭包(Closures) 是自包含的功能代码块,可以在代码中使用或者用来作为参数传值
从某些方面说,闭包有点类似匿名函数
全局函数和嵌套函数其实就是特殊的闭包
全局函数 | 嵌套函数 | 闭包表达式 |
---|---|---|
有名字但不能捕获任何值 | 有名字,也能捕获封闭函数内的值 | 无名闭包,使用轻量级语法,可以根据上下文环境捕获值 |
Swift 中的闭包有很多优化的地方
1、 可根据上下文推断参数和返回值类型;
2、 从单行表达式闭包中隐式返回(也就是闭包体只有一行代码,可以省略return);
3、 可以使用简化参数名,如 $
0, $
1(从0开始,表示第i个参数...);
4、 提供了尾随闭包语法(Trailingclosuresyntax);
下面代码定义了一个接收参数并返回指定类型的闭包语法:
{
(parameters) -> return type in
statements
}
import Cocoa
let techer_name = { print("DDKK.COM 弟弟快看,程序员编程资料站,DDKK.COM 弟弟快看,程序员编程资料站") }
techer_name()
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
DDKK.COM 弟弟快看,程序员编程资料站,DDKK.COM 弟弟快看,程序员编程资料站
下面的闭包形式接收两个参数并返回布尔值
{(Int, Int) -> Bool in
Statement1
Statement2
---
Statementn
}
import Cocoa
let mul = { (val1: Int, val2: Int) -> Int in
return val1 * val2
}
let result = mul(200, 20)
print (result)
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
4000
闭包表达式是一种利用简洁语法构建内联闭包的方式
闭包表达式提供了一些语法优化,让开发者定义闭包变得简单明了
Swift 标准库提供了名为 sorted(by:) 的方法,会根据提供的用于排序的闭包函数将已知类型数组中的值进行排序
排序完成后,sorted(by:) 方法会返回一个与原数组大小相同,包含同类型元素且元素已正确排序的新数组
原数组不会被 sorted(by:) 方法修改
sorted(by:)方法需要传入两个参数:
1、 已知类型的数组;
2、 闭包函数,该闭包函数需要传入与数组元素类型相同的两个值,并返回一个布尔类型值来表明当排序结束后传入的第一个参数排在第二个参数前面还是后面;
如果第一个参数值出现在第二个参数值前面,排序闭包函数需要返回 true ,反之返回 false
import Cocoa
let cops = ["BE", "BA", "C", "T", "BE"]
func sort_backwards(s1: String, s2: String) -> Bool {
return s1 > s2
}
var reversed = cops.sorted(by: sort_backwards)
print(reversed)
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["T", "C", "BE", "BE", "BA"]
如果第一个字符串 (s1) 大于第二个字符串 (s2),backwards 函数返回 true, 表示在新的数组中 s1 应该出现在 s2 前 对于字符串中的字符来说,"大于" 表示 "按照字母顺序较晚出现" 这意味着字母 "B" 大于字母 "A",字符串 "S" 大于字符串 "D"
其将进行字母逆序排序,"AT"将会排在"AE"之前
Swift 自动为内联函数提供了参数名称简写功能
我们可以直接通过 $
0, $
1, $
2 来顺序调用闭包的参数
import Cocoa
let cops = ["BE", "BA", "C", "T", "BE"]
var reversed = cops.sorted(by: { $0 > $1 })
print(reversed)
$
0 和 $
1 表示闭包中第一个和第二个 String 类型的参数
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["T", "C", "BE", "BE", "BA"]
如果在闭包表达式中使用参数名称缩写, 可以在闭包参数列表中省略对其定义, 并且对应参数名称缩写的类型会通过函数类型进行推断
in 关键字同样也可以被省略
实际上还有一种更简短的方式来撰写上面例子中的闭包表达式,就是使用 运算符重载
Swift 的 String 类型定义了关于大于号 ( >` ) 的字符串实现,其作为一个函数接受两个 String 类型的参数并返回 Bool 类型的值
而这正好与 sort(_:) 方法的第二个参数需要的函数类型相符合
所以我们可以简单地传递一个大于号,Swift可以自动推断出您想使用大于号的字符串函数实现:
import Cocoa
let cops = ["BE", "BA", "C", "T", "BE"]
var reversed = cops.sorted(by: > )
print(reversed)
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["T", "C", "BE", "BE", "BA"]
尾随闭包 是一个书写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用
func someFunctionThatTakesAClosure(closure: () -> Void) {
// 函数体部分
}
// 以下是不使用尾随闭包进行函数调用
someFunctionThatTakesAClosure({
// 闭包主体部分
})
// 以下是使用尾随闭包进行函数调用
someFunctionThatTakesAClosure() {
// 闭包主体部分
}
import Cocoa
let cops = ["BE", "BA", "C", "T", "BE"]
var reversed = cops.sorted(){ $0 > $1 }
print(reversed)
sort() 后的 { $
0 >$` 1} 为尾随闭包
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["T", "C", "BE", "BE", "BA"]
如果函数只需要闭包表达式一个参数,当您使用尾随闭包时,您甚至可以把()省略掉。
import Cocoa
let cops = ["BE", "BA", "C", "T", "BE"]
var reversed = cops.sorted{ $0 > $1 }
print(reversed)
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["T", "C", "BE", "BE", "BA"]
闭包可以在其定义的上下文中捕获常量或变量
即使定义这些常量和变量的原域已经不存在,闭包仍然可以在闭包函数体内引用和修改这些值
Swift 最简单的闭包形式是嵌套函数,也就是定义在其他函数的函数体内的函数 嵌套函数可以捕获其外部函数所有的参数以及定义的常量和变量
import Cocoa
func makeIncrementor(forIncrement amount: Int) -> () -> Int {
var runningTotal = 0
func incrementor() -> Int {
runningTotal += amount
return runningTotal
}
return incrementor
}
let incr = makeIncrementor(forIncrement:5)
print(incr())
print(incr())
print(incr())
一个函数 makeIncrementor ,它有一个 Int 型的参数 amout, 并且它有一个外部参数名字 forIncremet,意味着你调用的时候,必须使用这个外部名字 返回值是一个()->` Int的函数
函数体内,声明了变量 runningTotal 和一个函数 incrementor
incrementor 函数并没有获取任何参数,但是在函数体内访问了 runningTotal 和amount变量。这是因为其通过捕获在包含它的函数体内已经存在的runningTotal和amount变量而实现。
由于没有修改amount变量,incrementor实际上捕获并存储了该变量的一个副本,而该副本随着incrementor一同被存储
编译编译运行以上 Swift 范例,输出结果为
$ swift main.swift
5
10
15
在上面的实例中 incrementByTen 是常量,但是这些常量指向的闭包仍然可以增加其捕获的变量值
这是因为函数和闭包都是引用类型
无论您将函数/闭包赋值给一个常量还是变量,您实际上都是将常量/变量的值设置为对应函数/闭包的引用。 上面的例子中,incrementByTen指向闭包的引用是一个常量,而并非闭包内容本身
这意味着如果您将闭包赋值给了两个不同的常量/变量,两个值都会指向同一个闭包:
import Cocoa
func makeIncrementor(forIncrement amount: Int) -> () -> Int {
var runningTotal = 0
func incrementor() -> Int {
runningTotal += amount
return runningTotal
}
return incrementor
}
let incr = makeIncrementor(forIncrement:5)
let incr2 = incr
print(incr())
print(incr2())
print(incr())
print(incr2())
print(incr())
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
5
10
15
20
25
以下闭包函数在 javascript 中运行良好。 function generateNextNumber(startNumber) { var current = startNumber;
Swift的闭包(Closures)是一种将功能块和上下文整合并演示在代码中的一种手段。闭包可以捕获并存储其上下文中的变量和常量。与普遍存在于其他语言的匿名函数(如Python的lambda、Java
在本教程中,您将借助示例了解 JavaScript 闭包。 在了解闭包之前,您需要了解两个概念: 嵌套函数 返回函数 JavaScript 嵌套函数 在 JavaScript 中,一个函数也可
在本教程中,您将借助示例了解 JavaScript 闭包。 在了解闭包之前,您需要了解两个概念: 嵌套函数 返回函数 JavaScript 嵌套函数 在 JavaScript 中,一个函数也可
闭包介绍 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。 要理解闭包,首先必须理解Javascript特殊的变量作用域。 1.全局变量和局部变
这个问题已经有答案了: Methods in ES6 objects: using arrow functions (6 个回答) 已关闭 6 年前。 我已经在 stackoverflow 上到处查找
这个问题已经有答案了: How do JavaScript closures work? (86 个回答) 已关闭 9 年前。 我有一个关于 Javascript 闭包的简单问题: 给出了以下函数:
所以我有以下内容: Object a = data.getA(); Object b = data.getB(); Object c = data.getC(); // and so on 这些对象是
现在已经很晚了,我大脑中道格拉斯·克罗克福德居住的部分已经关闭。我尝试了一些方法,但没有达到预期效果。 我有一个 Canvas ,我在其中画了两条线,然后在计时器上淡出它们,但只有循环中的最后一行淡出
因此,我创建了一个变量 car,然后将其分配给一个函数并添加了参数模型、年份。然后在函数内引用参数创建一个对象。 然后创建“闭包”内部函数 yourCar() 并返回其中的外部函数对象“Propert
我正在 Mozilla 开发者网站上阅读关于关闭的解释,并且有点挣扎。请查看 Mozilla 网站上的以下代码。我有点理解它是如何工作的,但我认为我的评论下面的代码也应该工作。为什么一点击18、20就
这个问题在这里已经有了答案: UnboundLocalError trying to use a variable (supposed to be global) that is (re)assig
以下程序返回“本地”,根据我正在阅读的教程,它旨在演示闭包现象` 我不明白的是,为什么最后为了调用parentfunction,将其分配给变量“child”,然后调用“child”。 为什么只写 pa
我读到闭包末尾的()会立即执行。那么,这两者之间有什么区别。我在一些代码中看到了第一个用法。 谢谢。 for (var a=selectsomeobj(),i=0,len=a.length;i
代码如下 var collection = (function (){ var x = 0; return {
我仍然对 JavaScript 中的闭包概念感到困惑。我明白闭包是内部函数在母函数返回后访问在其母函数中创建的变量的能力。但是我仍然很困惑,如果我们可以在函数内部创建一个变量,为什么我们必须创建内部函
我搜索了很多主题并没有找到答案,或者问题太复杂了。所以没关系。这是我的第一个问题。 这是 SQL SELECT parent.*, ( SELECT COUNT(*) FROM
有 JS 高手可以解释为什么会这样吗: $$={} (function(x){ x.newModule = { func: function(){...} };
在此示例中,我尝试按值传递,但传递的是引用。 for (int i = 0; i new PhoneJobTest(i); t.Start(); } 这可以像这样补救: for (int
从 $.each() 中访问 this.rules 变量的最佳方式是什么?任何关于原因/方式的解释也会有帮助! app.Style = function(node) { this.style
我是一名优秀的程序员,十分优秀!