Swift学习Tip之Closures

一、闭包简介

亲,我的简书已不再维护和更新了,所有文章都迁移到了我的个人博客:https://mikefighting.github.io/,欢迎交流。

什么是ClosuresClosures也就是我们常说的闭包,闭包是一个具有函数功能的代码块,它类似于OC中的Block,它能够在它被定义的上下文中,被常量或者变量所捕捉和存储。也就是是说把常量和变量Closing(包)住了,这也是Closures的由来。全局函数和嵌套函数(Global and nested functions)是闭包的一种特例,闭包有一下三种形式:

  • 全局函数是有名称但是不会捕获任何数值得闭包。
  • 嵌套函数是有名称并且可以从它们的封闭的函数体内部捕获数值的闭包。
  • 闭包表达式是用轻量级的语言所书写,并且能够从它们所在的上下文中获取数值的一种没有名称的closures。

由于Swift本身对闭包做了很多的优化,这使得闭包的使用更加的简单明了,这些优化包括:

  • 从上下文中推断出参数类型和返回值类型
  • 在单一表达的闭包(single-expression)中隐含返回值
  • 简化参数名
  • 尾随闭包语法

二、闭包的基本语法

//闭包表达式的类型和函数的类型是一样的,都是参数加上返回值,但是闭包后面有一个in, 这个关键字来告诉编译器,闭包的参数和返回值已经定义完了,下面是闭包执行语句了。
{
(参数) -> 返回值类型 in
执行语句
}

下面说明闭包的基本用法:

  • 闭包的完整写法:
  // 闭包的完整写法
  let sayHiToSomeBody:(String) -> Void = {
            (name: String) -> Void in
            print("Hi \(name)")
        }
        sayHiToSomeBody("Mike")
        // Print: Hi Mike  
  • 没有返回值得闭包
        // 由于这个闭包没有返回值,所以Void可以省略
        let sayHiToSomeBodyWithoutReturnValue:(String) -> Void = {
            (name: String) in
            print("Hi \(name)")
        
        }
        sayHiToSomeBodyWithoutReturnValue("Jimme")        
  • 没有参数没有返回值得闭包
        // 没有参数没有返回值的写法
        let sayHiToSomeBodyWithoutParaAndReturnValue:() ->Void = {
          
          print("Hi Wallace")
          
        }
        sayHiToSomeBodyWithoutParaAndReturnValue() 
        // Print: Hi Wallace
  • 闭包做为一个参数
       // 闭包做为一个参数
        func functionToRunClosureAsParam(someClosure:() -> ()) ->String{
            
            print("run before the closure")
            someClosure()
            return "I ran the closure"
            
        }
        
        print(functionToRunClosureAsParam({print("try it now")}))
        // Print:
        // run before the closure
        // try it now
        // I ran the closure
  • 闭包做为函数的返回值

        func makeFunctionWithClosureToReturn(somePara:String) ->() -> (){ // 闭包做为函数的返回值
            
            
            let resultClosure = { print("\(somePara)")}
            return resultClosure
        }
        
        let yellSomeString = makeFunctionWithClosureToReturn("I Love Swift")
        yellSomeString() 
        // Print: I Love Swift
        
  • 捕获值,(在闭包内部捕获闭包所在的上下文(surrounding context)的变量)
        // 捕获值(这里也是一个以闭包做为返回值的函数)
        func someFuncWithClosureReturn(forIncrement amount: Int) -> () ->Int{
            
            var runningTotal = 0
            // 这个闭包是没有参数,只有返回值,这个返回值是由someFuncWithClosureReturn()的参数,和内部的变量
            let incrementer:()->Int = {
                
                runningTotal += amount
                return runningTotal
                
            }
            //            函数其实也是闭包的一种,所以可以这样写
            //            func incrementer() ->Int {
            //
            //                runningTotal += amount;
            //                return runningTotal
            //
            //            }
            return incrementer
        }
        let incrementByTen = someFuncWithClosureReturn(forIncrement: 10)
        
        print(incrementByTen()) // Print 10
        print(incrementByTen()) // Print 20
        print(incrementByTen()) // Print 30
  • 尾随闭包Trailing Closures:如果闭包是函数的最后一个参数,并且闭包比较长,这个时候可以在()的后面写一个{}来做为闭包。上文提到的functionToRunClosureAsParam()就是这样一个例子,所以我们可以使用尾随闭包来简化函数的调用。
       func functionToRunClosureAsParam(someClosure:() -> ()) ->String //闭包做为函数的最后一个参数
            
            print("run before the closure")
            someClosure()
            return "I ran the closure"
            
        }
        print(functionToRunClosureAsParam(){print("Try it now ")}) // 这里将闭包提出了()
        // 而不是上文中的:print(functionToRunClosureAsParam({print("try it now")}))

三、闭包的高级用法

  • Noescape Closures:非逃逸型闭包,如果闭包是函数的一个参数,那么这个闭包只能在函数内部使用;Escape Closures:逃逸型闭包,当闭包做为函数的一个参数的时候,有时候在函数已经执行完,并且已经返回,但是这之后才调用闭包,这种闭包就称为逃逸闭包(Escaping Closure),逃逸型闭包可以用在异步操作。
       // 非逃逸型闭包
        func someFunctionWithNoescapeClosure(@noescape closure: () -> Void) {
            closure()
        }
        
        // 逃逸型闭包,如果强制将其改变为非逃逸型的将会报错,因为非逃逸型只能在函数内部使用,而这个闭包却被存在了函数之外的数组中,所以必须是逃逸型的,如果强制改为非逃逸型的,会报错
        var completionHandlers: [() -> Void] = [] // 这是一个存贮闭包的数组
        func someFunctionWithEscapingClosure(completionHandler: () -> Void) {
            completionHandlers.append(completionHandler)
        }
        
        // 非逃逸型的闭包如果使用了全局变量,那么可以省去self,下面举例说明
        class SomeClass {
            var x = 10
            func doSomething() {
                someFunctionWithEscapingClosure { self.x = 100 }
                someFunctionWithNoescapeClosure { x = 200 }
            }
        }

        let instance = SomeClass()
        instance.doSomething()
        print(instance.x)
        // prints "200"
        
        completionHandlers.first?() // 这个时候取出数组中存放的已经逃逸的闭包,然后执行它,就会让 instanc.x变为100
        print(instance.x)
        // prints "100”

下面在举个例子说明Noescape Closures的用法:

        // 再比如非逃逸型闭包
        func doIt(@noescape code: () -> ()) {
            /* 我们可以这样做 */
            // 仅调用它
            code()
            // 做为非逃逸型参数传递给另外一个函数
            doItMore(code)
            // 在另外一个非逃逸的函数中捕获智
            doItMore {
                code()
            }
            /* 我们不可以做 */
            
            /*
            // 将它赋值为一个需要可逃逸闭包参数的函数
            dispatch_async(dispatch_get_main_queue(), code)
            // 存储它
            let _code:() -> () = code
            // 在另一个可逃逸的闭包
            let __code = { code() }
            
            */
        }
        
        func doItMore(@noescape code: () -> ()) {}
  • Autoclosures自动闭包:如果函数中的一个闭包参数被@autoclosures修饰,那么这个函数在被调用的时候可以省去花括号,注意如果一个闭包被autoClosures修饰的时候默认是noescape型的,所以在不能用到异步处理中,这是只需要在将@autoclosure变为@autoclosure(escaping)就可以了

下面举例说明:

        // customersInLine is ["Ewa", "Barry", "Daniella"]
        func serveCustomer(@autoclosure customerProvider: () -> String) {
            print("Now serving \(customerProvider())!")
        }
        serveCustomer(customersInLine.removeAtIndex(0)) // 这个时候回省去{}
        
        // 再比如一个以闭包做为参数的函数
        func f(@autoclosure pred: () -> Bool) {
            if pred() {
                print("It's true")
            }
        }
        f(2 > 1) // 如果不加@autoclosure,我们不得不f({2 > 1})
        // It's true
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容

  • 闭包是自包含的函数代码块,可以在代码中被传递和使用。Swift 中的闭包与 C 和 Objective-C 中的代...
    莽原奔马668阅读 1,870评论 2 12
  • 闭包是自包含的函数代码块,可以在代码中被传递和使用。Swift 中的闭包与 C 和 Objective-C 中的代...
    穷人家的孩纸阅读 1,690评论 1 5
  • 闭包是自包含的函数代码块,可以在代码中被传递和使用。Swift 中的闭包与 C 和 Objective-C 中的代...
    雨影阅读 714评论 0 2
  • 86.复合 Cases 共享相同代码块的多个switch 分支 分支可以合并, 写在分支后用逗号分开。如果任何模式...
    无沣阅读 1,327评论 1 5
  • 其实,读书和读大学是两回事儿 !读大学的目的,更多的是在于自身内在心性与外在气质的修炼,或许这也正是腹有诗书气自华...
    金婉豆阅读 362评论 0 1