Swift教學
Swift快速入門
Swift簡介
Swift基礎
Swift常量和變量
Swift註釋
swift分號
Swift整數
Swift浮點數
Swift類型安全和類型推斷
swift數值型字面量
Swift數值型類型轉換
Swift類型別名
Swift布爾值
Swift元組
Swift可選類型
Swift斷言
Swift基本運算符
Swif哈希集合
Swift運算術語
Swift字典
Swift賦值運算符
Swift數值運算
Swift複合賦值
Swift比較運算
Swift三元條件運算
Swift區間運算符
Swift邏輯運算
Swift字符串和字符
Swift字符串字面量
Swift初始化空字符串
Swift字符串可變性
Swift字符串是值類型
Swift使用字符
Swift計算字符數量
Swift連接字符串和字符
Swift字符串插值
Swift比較字符串
Swift大寫和小寫字符串
Swift Unicode
Swift集合類型 (Collection Types)
Swift數組
Swift字典
Swift集合的可變性
Swift控制流
Swift For循環
Swift While循環
Swift條件語句
Swift控制轉移語句
Swift函數
Swift函數的定義與調用
Swift函數參數與返回值
Swift函數參數名稱
Swift函數類型
Swift嵌套函數
Swift閉包
Swift閉包表達式
Swift尾隨閉包
Swift捕獲值
Swift閉包是引用類型
Swift枚舉
Swift枚舉語法
Swift匹配枚舉值和Switch語句
Swift相關值
Swift原始值
Swift類和結構體
Swift類和結構體對比
Swift結構體和枚舉是值類型
Swift類是引用類型
Swift類和結構體的選擇
Swift集合(Collection)類型的賦值和拷貝行爲
Swift屬性
Swift存儲屬性
Swift計算屬性
Swift屬性監視器
Swift全局變量和局部變量
Swift類型屬性
Swift方法
Swift實例方法
Swift類型方法
Swift下標腳本
Swift下標腳本語法
Swift下標腳本用法
Swift下標腳本選項
Swift繼承
Swift定義一個基類
Swift子類生成
Swift重寫
Swift防止重寫
Swift構造過程
Swift存儲型屬性的初始賦值
Swift定製化構造過程
Swift默認構造器
Swift值類型的構造器代理
Swift類的繼承和構造過程
Swift通過閉包和函數來設置屬性的默認值
Swift析構過程
Swift析構過程原理
Swift析構函數操作
Swift自動引用計數
Swift自動引用計數的工作機制
Swift自動引用計數實踐
Swift類實例之間的循環強引用
Swift解決實例之間的循環強引用
Swift閉包引起的循環強引用
Swift解決閉包引起的循環強引用
Swift可選鏈
Swift可選鏈可替代強制解析
Swift爲可選鏈定義模型類
Swift通過可選鏈調用屬性
Swift通過可選鏈調用方法
Swift使用可選鏈調用子腳本
Swift連接多層鏈接
Swift鏈接可選返回值的方法
Swift類型轉換
Swift定義一個類層次作爲例子
Swift檢查類型
Swift向下轉型
Swift Any和AnyObject類型轉換
Swift嵌套類型
Swift嵌套類型實例
Swift嵌套類型的引用
Swift擴展
Swift擴展語法
Swift計算型屬性
Swift構造器
Swift方法擴展
Swift下標
Swift嵌套類型擴展
Swift協議
Swift協議的語法
Swift屬性要求
Swift方法要求
Swift突變方法要求
Swift協議類型
Swift委託(代理)模式
Swift在擴展中添加協議成員
Swift通過擴展補充協議聲明
Swift集合中的協議類型
Swift協議的繼承
Swift協議合成
Swift檢驗協議的一致性
Swift可選協議要求
Swift泛型
Swift泛型所解決的問題
Swift泛型函數
Swift類型參數
Swift命名類型參數
Swift泛型類型
Swift類型約束
Swift關聯類型
Swift Where語句
Swift高級運算符
Swift位運算符
Swift溢出運算符
Swift優先級和結合性
Swift運算符函數
Swift自定義運算符
Swift語法結構
Swift空白與註釋
Swift標識符
Swift關鍵字
Swift字面量
Swift運算符
Swift類型
Swift類型註解
Swift類型標識符
Swift元組類型
Swift函數類型(參數類型和返回值類型)
Swift數組類型
Swift可選類型(命名型類型)
Swift隱式解析可選類型
Swift協議合成類型
Swift元類型
Swift類型繼承子句
Swift類型推斷
Swift表達式
Swift前綴表達式
Swift二元表達式
Swift賦值表達式
Swift類型轉換運算符
Swift主表達式
Swift後綴表達式
Swift語句
Swift循環語句
Swift For語句
Swift分支語句
Swift帶標籤的語句
Swift聲明
Swift模塊範圍
Swift代碼塊
Swift引入聲明
Swift常量聲明
Swift類型的別名聲明
Swift函數聲明
Swift枚舉聲明
Swift結構體聲明
Swift類聲明
Swift協議聲明
Swift構造器聲明
Swift析構聲明
Swift擴展聲明
Swift下標腳本聲明
Swift運算符聲明
Swift變量聲明
Swift特性
Swift聲明特性
Swift類型特性
Swift模式
Swift通配符模式
Swift標識符模式
Swift值綁定模式
Swift元組模式
Swift枚舉用例模式
Swift類型轉換模式
Swift表達式模式
Swift泛型參數
Swift泛型形參子句
Swift開發環境設置
Swift基本語法
Swift數據類型
Swift變量
Swift常量
Swift字面量
Swift運算符
Swift比較運算符
Swift邏輯運算符
Swift位運算符
Swift賦值運算符
Swift範圍運算符
Swift其它運算符
Swift運算符優先級
Swift算術運算符
Swift if語句
Swift if...else語句
Swift if...else if...else語句
Swift嵌套 if 語句
Swift Switch語句
Swift決策
Swift for-in循環
Swift for循環
Swift while循環
Swift do...while循環
Swift continue語句
Swift break語句
Swift fallthrough語句
Swift循環
Swift字符串
Swift字符
Swift數組
Swift函數
Swift閉包
Swift枚舉
Swift結構體
Swift類
Swift 屬性
Swift 方法
Swift 下標
Swift 繼承
Swift初始化
Swift 反初始化
Swift ARC自動引用計數
Swift 可選鏈
Swift 類型轉換
Swift 擴展
Swift 協議
Swift 泛型
Swift訪問控制

Swift閉包表達式

閉包表達式(Closure Expressions)

嵌套函數 是一個在較複雜函數中方便進行命名和定義自包含代碼模塊的方式。當然,有時候撰寫小巧的沒有完整定義和命名的類函數結構也是很有用處的,尤其是在您處理一些函數並需要將另外一些函數作爲該函數的參數時。

閉包表達式是一種利用簡潔語法構建內聯閉包的方式。 閉包表達式提供了一些語法優化,使得撰寫閉包變得簡單明瞭。 下面閉包表達式的例子通過使用幾次迭代展示了sort函數定義和語法優化的方式。 每一次迭代都用更簡潔的方式描述了相同的功能。

sort 函數(The Sort Function)

Swift 標準庫提供了sort函數,會根據您提供的基於輸出類型排序的閉包函數將已知類型數組中的值進行排序。 一旦排序完成,函數會返回一個與原數組大小相同的新數組,該數組中包含已經正確排序的同類型元素。

下面的閉包表達式示例使用sort函數對一個String類型的數組進行字母逆序排序,以下是初始數組值:

let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]

sort函數需要傳入兩個參數:

  • 已知類型的數組
  • 閉包函數,該閉包函數需要傳入與數組類型相同的兩個值,並返回一個布爾類型值來告訴sort函數當排序結束後傳入的第一個參數排在第二個參數前面還是後面。如果第一個參數值出現在第二個參數值前面,排序閉包函數需要返回true,反之返回false

該例子對一個String類型的數組進行排序,因此排序閉包函數類型需爲(String, String) -> Bool

提供排序閉包函數的一種方式是撰寫一個符合其類型要求的普通函數,並將其作爲sort函數的第二個參數傳入:

func backwards(s1: String, s2: String) -> Bool {
    return s1 > s2
}
var reversed = sort(names, backwards)
// reversed 爲 ["Ewa", "Daniella", "Chris", "Barry", "Alex"]

如果第一個字符串 (s1) 大於第二個字符串 (s2),backwards函數返回true,表示在新的數組中s1應該出現在s2前。 對於字符串中的字符來說,「大於」 表示 「按照字母順序較晚出現」。 這意味着字母"B"大於字母"A",字符串"Tom"大於字符串"Tim"。 其將進行字母逆序排序,"Barry"將會排在"Alex"之後。

然而,這是一個相當冗長的方式,本質上只是寫了一個單表達式函數 (a > b)。 在下面的例子中,利用閉合表達式語法可以更好的構造一個內聯排序閉包。

閉包表達式語法(Closure Expression Syntax)

閉包表達式語法有如下一般形式:

{ (parameters) -> returnType in
    statements
}

閉包表達式語法可以使用常量、變量和inout類型作爲參數,不提供默認值。 也可以在參數列表的最後使用可變參數。 元組也可以作爲參數和返回值。

下面的例子展示了之前backwards函數對應的閉包表達式版本的代碼:

reversed = sort(names, { (s1: String, s2: String) -> Bool in
    return s1 > s2
})

需要注意的是內聯閉包參數和返回值類型聲明與backwards函數類型聲明相同。 在這兩種方式中,都寫成了(s1: String, s2: String) -> Bool。 然而在內聯閉包表達式中,函數和返回值類型都寫在大括號內,而不是大括號外。

閉包的函數體部分由關鍵字in引入。 該關鍵字表示閉包的參數和返回值類型定義已經完成,閉包函數體即將開始。

因爲這個閉包的函數體部分如此短以至於可以將其改寫成一行代碼:

reversed = sort(names, { (s1: String, s2: String) -> Bool in return s1 > s2 } )

這說明sort函數的整體調用保持不變,一對圓括號仍然包裹住了函數中整個參數集合。而其中一個參數現在變成了內聯閉包(相比於backwards版本的代碼)。

根據上下文推斷類型(Inferring Type From Context)

因爲排序閉包函數是作爲sort函數的參數進行傳入的,Swift可以推斷其參數和返回值的類型。 sort期望第二個參數是類型爲(String, String) -> Bool的函數,因此實際上String,StringBool類型並不需要作爲閉包表達式定義中的一部分。 因爲所有的類型都可以被正確推斷,返回箭頭 (->) 和圍繞在參數周圍的括號也可以被省略:

reversed = sort(names, { s1, s2 in return s1 > s2 } )

實際上任何情況下,通過內聯閉包表達式構造的閉包作爲參數傳遞給函數時,都可以推斷出閉包的參數和返回值類型,這意味着您幾乎不需要利用完整格式構造任何內聯閉包。

單表達式閉包隱式返回(Implicit Return From Single-Expression Clossures)

單行表達式閉包可以通過隱藏return關鍵字來隱式返回單行表達式的結果,如上版本的例子可以改寫爲:

reversed = sort(names, { s1, s2 in s1 > s2 } )

在這個例子中,sort函數的第二個參數函數類型明確了閉包必須返回一個Bool類型值。 因爲閉包函數體只包含了一個單一表達式 (s1 > s2),該表達式返回Bool類型值,因此這裏沒有歧義,return關鍵字可以省略。

參數名稱縮寫(Shorthand Argument Names)

Swift 自動爲內聯函數提供了參數名稱縮寫功能,您可以直接通過$0,$1,$2來順序調用閉包的參數。

如果您在閉包表達式中使用參數名稱縮寫,您可以在閉包參數列表中省略對其的定義,並且對應參數名稱縮寫的類型會通過函數類型進行推斷。 in關鍵字也同樣可以被省略,因爲此時閉包表達式完全由閉包函數體構成:

reversed = sort(names, { $0 > $1 } )

在這個例子中,$0$1表示閉包中第一個和第二個String類型的參數。

運算符函數(Operator Functions)

實際上還有一種更簡短的方式來撰寫上面例子中的閉包表達式。 Swift 的String類型定義了關於大於號 (>) 的字符串實現,其作爲一個函數接受兩個String類型的參數並返回Bool類型的值。 而這正好與sort函數的第二個參數需要的函數類型相符合。 因此,您可以簡單地傳遞一個大於號,Swift可以自動推斷出您想使用大於號的字符串函數實現:

reversed = sort(names, >)

更多關於運算符表達式的內容請查看 運算符函數。