swift學習筆記4——擴展、協議

来源:http://www.cnblogs.com/mddblog/archive/2016/06/21/5604914.html
-Advertisement-
Play Games

之前學習swift時的個人筆記,根據github: "the swift programming language in chinese" 學習、總結,將重要的內容提取,加以理解後整理為學習筆記,方便以後查詢用。詳細可以參考 "the swift programming language in ch ...


之前學習swift時的個人筆記,根據github:the-swift-programming-language-in-chinese學習、總結,將重要的內容提取,加以理解後整理為學習筆記,方便以後查詢用。詳細可以參考the-swift-programming-language-in-chinese,或者蘋果官方英文版文檔

當前版本是swift2.2

擴展(Extensions)

擴展 就是為一個已有的類、結構體、枚舉類型或者協議類型添加新功能。這包括在沒有許可權獲取原始源代碼的情況下擴展類型的能力(即 逆向建模 )。擴展和 Objective-C 中的分類類似。(與 Objective-C 不同的是,Swift 的擴展沒有名字。)

如果你通過擴展為一個已有類型添加新功能,那麼新功能對該類型的所有已有實例都是可用的,即使它們是在這個擴展定義之前創建的。

計算型屬性(Computed Properties)

擴展可以為已有類型添加計算型實例屬性和計算型類型屬性。不可以添加存儲性屬性

構造器(Initializers)

擴展可以為已有類型添加新的構造器。這可以讓你擴展其它類型,將你自己的定製類型作為其構造器參數,或者提供該類型的原始實現中未提供的額外初始化選項。

擴展能為類添加新的便利構造器,但是它們不能為類添加新的指定構造器或析構器。指定構造器和析構器必須總是由原始的類實現來提供。

方法(Methods)

擴展可以為已有類型添加新的實例方法和類型方法。

可變實例方法(Mutating Instance Methods)

通過擴展添加的實例方法也可以修改該實例本身。結構體和枚舉類型中修改 self 或其屬性的方法必須將該實例方法標註為 mutating,正如來自原始實現的可變方法一樣。

下麵的例子為 Swift 的 Int 類型添加了一個名為 square 的可變方法,用於計算原始值的平方值:

extension Int {
    mutating func square() {
        self = self * self
    }
}

下標(Subscripts)

擴展可以為已有類型添加新下標。這個例子為 Swift 內建類型 Int 添加了一個整型下標。該下標 [n] 返回十進位數字從右向左數的第 n 個數字:

123456789[0] 返回 9
123456789[1] 返回 8

嵌套類型(Nested Types)

擴展可以為已有的類、結構體和枚舉添加新的嵌套類型:

協議

Mutating 方法要求

如果你在協議中定義了一個實例方法,該方法會改變採納該協議的類型的實例,那麼在定義協議時需要在方法前加 mutating 關鍵字。這使得結構體和枚舉能夠採納此協議並滿足此方法要求。

將 mutating 關鍵字作為方法的首碼,寫在 func 關鍵字之前,表示可以在該方法中修改它所屬的實例以及實例的任意屬性的值

實現協議中的 mutating 方法時,若是類類型,則不用寫 mutating 關鍵字。而對於結構體和枚舉,則必須寫 mutating 關鍵字。

協議構造器要求

協議可以要求採納協議的類型實現指定的構造器。你可以像編寫普通構造器那樣,在協議的定義里寫下構造器的聲明,但不需要寫花括弧和構造器的實體:

protocol SomeProtocol {
    init(someParameter: Int)
}

構造器要求在類中的實現

你可以在採納協議的類中實現構造器,無論是作為指定構造器,還是作為便利構造器。無論哪種情況,你都必須為構造器實現標上 required 修飾符:

class SomeClass: SomeProtocol {
    required init(someParameter: Int) {
        // 這裡是構造器的實現部分
    }
}

使用 required 修飾符可以確保所有子類也必須提供此構造器實現,從而也能符合協議。

構造器要求
協議可以要求採納協議的類型實現指定的構造器。你可以像編寫普通構造器那樣,在協議的定義里寫下構造器的聲明,但不需要寫花括弧和構造器的實體:

protocol SomeProtocol {
init(someParameter: Int)
}
構造器要求在類中的實現

你可以在採納協議的類中實現構造器,無論是作為指定構造器,還是作為便利構造器。無論哪種情況,你都必須為構造器實現標上 required 修飾符:

class SomeClass: SomeProtocol {
    required init(someParameter: Int) {
        // 這裡是構造器的實現部分
    }
}

使用 required 修飾符可以確保所有子類也必須提供此構造器實現,從而也能符合協議。

如果一個子類重寫了父類的指定構造器,並且該構造器滿足了某個協議的要求,那麼該構造器的實現需要同時標註 required 和 override 修飾符:

protocol SomeProtocol {
    init()
}

class SomeSuperClass {
    init() {
        // 這裡是構造器的實現部分
    }
}

class SomeSubClass: SomeSuperClass, SomeProtocol {
    // 因為採納協議,需要加上 required
    // 因為繼承自父類,需要加上 override
    required override init() {
        // 這裡是構造器的實現部分
    }
}

如果類已經被標記為 final,那麼不需要在協議構造器的實現中使用 required 修飾符,因為 final 類不能有子類

協議作為類型

儘管協議本身並未實現任何功能,但是協議可以被當做一個成熟的類型來使用。

  • 作為函數、方法或構造器中的參數類型或返回值類型
  • 作為常量、變數或屬性的類型
  • 作為數組、字典或其他容器中的元素類型

通過擴展採納協議

當一個類型已經符合了某個協議中的所有要求,卻還沒有聲明採納該協議時,可以通過空擴展體的擴展來採納該協議:

struct Hamster {
    var name: String
    var textualDescription: String {
        return "A hamster named \(name)"
    }
}
extension Hamster: TextRepresentable {}

從現在起,Hamster 的實例可以作為 TextRepresentable 類型使用:

let simonTheHamster = Hamster(name: "Simon")
let v1: TextRepresentable = simonTheHamster // 可以賦值
print(v1.textualDescription)

即使滿足了協議的所有要求,類型也不會自動採納協議,必須顯式地採納協議。

協議類型的集合

協議類型可以在數組或者字典這樣的集合中使用,在協議類型提到了這樣的用法。下麵的例子創建了一個元素類型為 TextRepresentable 的數組:

let things: [TextRepresentable] = [game, d12, simonTheHamster]
如下所示,可以遍歷 things 數組,並列印每個元素的文本表示:

for thing in things {
    print(thing.textualDescription)
}
// A game of Snakes and Ladders with 25 squares
// A 12-sided dice
// A hamster named Simon

thing 是 TextRepresentable 類型而不是 Dice,DiceGame,Hamster 等類型,即使實例在幕後確實是這些類型中的一種。由於 thing 是 TextRepresentable 類型,任何 TextRepresentable 的實例都有一個 textualDescription 屬性,所以在每次迴圈中可以安全地訪問 thing.textualDescription。

協議的繼承

協議能夠繼承一個或多個其他協議,可以在繼承的協議的基礎上增加新的要求。協議的繼承語法與類的繼承相似,多個被繼承的協議間用逗號分隔:

protocol InheritingProtocol: SomeProtocol, AnotherProtocol {
    // 這裡是協議的定義部分
}

類類型專屬協議

你可以在協議的繼承列表中,通過添加 class 關鍵字來限制協議只能被類類型採納,而結構體或枚舉不能採納該協議。class 關鍵字必須第一個出現在協議的繼承列表中,在其他繼承的協議之前:

protocol SomeClassOnlyProtocol: class, SomeInheritedProtocol {
    // 這裡是類類型專屬協議的定義部分
}

在以上例子中,協議 SomeClassOnlyProtocol 只能被類類型採納。如果嘗試讓結構體或枚舉類型採納該協議,則會導致編譯錯誤。

協議合成

有時候需要同時採納多個協議,你可以將多個協議採用 protocol 這樣的格式進行組合,稱為 協議合成(protocol composition)。你可以在 <> 中羅列任意多個你想要採納的協議,以逗號分隔。

下麵的例子中,將 Named 和 Aged 兩個協議按照上述語法組合成一個協議,作為函數參數的類型:

protocol Named {
    var name: String { get }
}
protocol Aged {
    var age: Int { get }
}
struct Person: Named, Aged {
    var name: String
    var age: Int
}
func wishHappyBirthday(celebrator: protocol<Named, Aged>) {
    print("Happy birthday \(celebrator.name) - you're \(celebrator.age)!")
}

協議合成並不會生成新的、永久的協議類型,而是將多個協議中的要求合成到一個只在局部作用域有效的臨時協議中。

可選的協議要求

協議可以定義可選要求,採納協議的類型可以選擇是否實現這些要求。在協議中使用 optional 關鍵字作為首碼來定義可選要求。使用可選要求時(例如,可選的方法或者屬性),它們的類型會自動變成可選的。比如,一個類型為 (Int) -> String 的方法會變成 ((Int) -> String)?。需要註意的是整個函數類型是可選的,而不是函數的返回值。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、日期格式轉時間戮 2、時間戮轉日期格式 ...
  • 經常我們看電影都會看到電影結束後會出現一段介紹這個電影的導演、編劇、主演、友情出演等等一些電影信息的滾動字幕,那麼那個效果可以用html+css+div實現。具體實現代碼如下: html代碼: 滾動標簽:<marquee>要滾動的文字/圖像<img src=””> </marquee> 下麵是我對滾 ...
  • 1,給第一人稱控制器添加腳本:playercollisions.js 腳本中只定義變數,先不添加方法: #pragma strict var door_open_time:float=3.0; var door_open_sound:AudioClip; var door_shut_sound:Au ...
  • 最近在做JS演算法項目時發現一個令我匪夷所思的問題, 這裡想記錄一下問題。 首先介紹一下字元串replace()方法的基本用法。 replace() 方法使用一個替換值(replacement)替換掉一個匹配模式(pattern)在原字元串中某些或所有的匹配項,並返回替換後的字元串。這個替換模式可以是 ...
  • App Store: Pinyin Comparison 拼音辨別 本人不懂韓文和日文,靠的是谷歌翻譯,希望不要被噴 1. 支持韓文和日文 2. 豐富字詞庫 1. 지원 한국과 일본 2. 풍부한 단어의 동의어 사전 1.サポート韓國と日本 2.豊富な単語シソーラス App Store: Piny ...
  • 一,效果圖。 二,工程圖。 三,代碼。 RootViewController.h #import <UIKit/UIKit.h> @interface RootViewController : UIViewController <UITextViewDelegate> { UITextView *p ...
  • 通常情況下,只有高端用戶才會經常用到終端應用。這並不意味著命令行非常難學,有的時候命令行可以輕鬆、快速的解決問題。相信所有Mac用戶都嘗試過命令行,今天為大家帶來9個非常實用的命令行操作。一些命令行需要安裝Xcode之後才可以實用,Xcode在Mac App Store中免費供應。1.使用caffe ...
  • 1.Android控制項架構下圖是UI界面架構圖,每個Activity都有一個Window對象,通常是由PhoneWindow類來實現的。PhoneWindow將DecorView作為整個應用視窗的根View,DecorView將屏幕分成兩部分:TitleView和ContentView。Conten ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...