Swift closure闭包

闭包.png

究竟什么是Closure?

说的通俗一点,一个函数加上它捕获的变量一起,才算一个closure

//MARK: - 闭包等于特殊的函数closure expression就是函数的一种简写形式
简单来说,closure expression就是函数的一种简写形式。例如,对于下面这个计算参数平方的函数:

func add(num1: Int , num2 : Int) -> Int{
    return  num1 * num2
}
//MARK: -  声明了方法
var addExp :(Int,Int)-> (Int) = { (a,b) -> (Int) in
    return a * b
}

//MARK: - 用于定义addExpression的{}就叫做closure expression,它只是把函数参数、返回值以及实现统统写在了一个{}里。

let addExpression = { (m: Int ,n: Int) -> Int in
    return m * n
}
func exec(m: Int, n: Int ,result: (Int ,Int) -> (Int)) -> Int{
    return result(m,n)
}
add(num1: 2, num2: 4)

addExp(2,4)

addExpression(2,4)
exec(m: 2, n: 4) { (a:Int, b: Int) -> Int in
    return a * b
}

//MARK: - 什么是尾随闭包 将闭包信息写在第一个

func exect(result: (Int ,Int) -> (Int) ,m: Int, n: Int ) -> Int{
    return result(m,n)
}

//MARK: - 如果一个函数的参数,是一个闭包类型,那么默认情况下,是一个非逃逸的闭包。

exect(result: { (a, b) -> (Int) in
    return a * b
}, m: 2, n: 4)

//MARK: - @escaping 代表这个闭包是逃逸闭包,以后有可能被其他的闭包所使用,延长生命周期

func test(result: @escaping (Int ,Int) ->(Int)){
    result(2,4)
    
    let queue = DispatchQueue.init(label: "result")
    let time = DispatchTime.now() + DispatchTimeInterval.seconds(2)
    queue.asyncAfter(deadline: time) { 
        result(2,4)
    }
}
func square(n: Int) -> Int {
    return n * n
}

我们也可以这样来定义:

let squareExpression = { (n: Int) -> Int in 
    return n * n
}

并且,它们也都可以当作函数参数来使用:

let numbers = [1, 2, 3, 4, 5]
numbers.map(square) // [1, 4, 9, 16, 25]
numbers.map(squareExpressions) // [1, 4, 9, 16, 25]

在我们这个例子里,用于定义squareExpression的{}就叫做closure expression,它只是把函数参数、返回值以及实现统统写在了一个{}里。

没错,此时的{}以及squareExpression并不能叫closure,它只是一个closure expression。那么,为什么要有两种不同的方式来定义函数呢?最直接的理由就是,为了写起来更简单。Closure expression可以在定义它的上下文里,被不断简化,让代码尽可能呈现出最自然的语义形态。

例如,当我们把一个完整的closure expression定义在map参数里,是这样的:

numbers.map({ (n: Int) -> Int in 
    return n * n
})

首先,Swift可以根据numbers的类型,自动推导出map中的函数参数以及返回值的类型,因此,我们可以在closure expression中去掉它:

numbers.map({ n in return n * n })

其次,如果closure expression中只有一条语句,Swift可以自动把这个语句的值作为整个expression的值返回,因此,我们还可以去掉return关键字:

numbers.map({ n in n * n })

第三,如果你觉得在closure expression中为参数起名字是个意义不大的事情,我们还可以使用Swift内置的$0/1/2/3/4这样的形式作为closure expression的参数替代符,这样,我们连参数声明和in关键字也都可以省略了:

numbers.map({ $0 * $0 })

第四,如果函数类型的参数在参数列表的最后一个,我们还可以把closure expression写在()外面,让它和其它普通参数更明显的区分开:

numbers.map() { $0 * $0 }

最后,如果函数只有一个函数类型的参数,我们甚至可以在调用的时候,去掉():

numbers.map { $0 * $0 }

看到这里,你就应该知道当我们把closure expression用在它的上下文里,究竟有多方便了,相比一开始的定义,或者单独定义一个函数,然后传递给它,都好太多。但事情至此还没结束,相比这样:

numbers.sorted(by: { $0 > $1 }) // [5, 4, 3, 2, 1]

Closure expression还有一种更简单的形式:

numbers.sorted(by: >) // [5, 4, 3, 2, 1]

这是因为,numbers.sorted(by:)的函数参数是这样的:(Int, Int) -> Bool,而Swift为Int类型定义的>操作符也正好和这个类型相同,这样,我们就可以直接把操作符传递给它,本质上,这和我们传递函数名是一样的。

另外,除了写起来更简单之外,closure expression还有一个副作用,就是默认情况下,我们无法忽略它的参数,编译器会对这种情况报错。来看个例子,如果我们要得到一个包含10个随机数的Array,最简单的方法,就是对一个CountableRange调用map方法:

(0...9).map { arc4random() } // !!! Error in swift !!!
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,192评论 6 511
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,858评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,517评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,148评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,162评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,905评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,537评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,439评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,956评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,083评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,218评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,899评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,565评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,093评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,201评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,539评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,215评论 2 358

推荐阅读更多精彩内容