Swift - 擴(kuò)展

擴(kuò)展


擴(kuò)展可以給一個(gè)現(xiàn)有的類掉缺,結(jié)構(gòu)體态秧,枚舉,還有協(xié)議添加新的功能店煞。它還擁有不需要訪問被擴(kuò)展類型源代碼就能完成擴(kuò)展的能力(即逆向建模)。擴(kuò)展和 Objective-C 的分類很相似风钻。(與 Objective-C 分類不同的是顷蟀,Swift 擴(kuò)展是沒有名字的。)

Swift 中的擴(kuò)展可以:

  • 添加計(jì)算型實(shí)例屬性和計(jì)算型類屬性

  • 定義實(shí)例方法和類方法

  • 提供新的構(gòu)造器

  • 定義下標(biāo)

  • 定義和使用新的嵌套類型

  • 使已經(jīng)存在的類型遵循(conform)一個(gè)協(xié)議

在 Swift 中骡技,你甚至可以擴(kuò)展協(xié)議以提供其需要的實(shí)現(xiàn)衩椒,或者添加額外功能給遵循的類型所使用。你可以從 協(xié)議擴(kuò)展 獲取更多細(xì)節(jié)哮兰。

注意

擴(kuò)展可以給一個(gè)類型添加新的功能毛萌,但是不能重寫已經(jīng)存在的功能。

擴(kuò)展的語法

使用 extension 關(guān)鍵字聲明擴(kuò)展:

extension SomeType {
  // 在這里給 SomeType 添加新的功能
}

擴(kuò)展可以擴(kuò)充一個(gè)現(xiàn)有的類型喝滞,給它添加一個(gè)或多個(gè)協(xié)議阁将。協(xié)議名稱的寫法和類或者結(jié)構(gòu)體一樣:

extension SomeType: SomeProtocol, AnotherProtocol {
  // 協(xié)議所需要的實(shí)現(xiàn)寫在這里
}

這種遵循協(xié)議的方式在 在擴(kuò)展里添加協(xié)議遵循 中有描述。

擴(kuò)展可以使用在現(xiàn)有范型類型上右遭,就像 泛型擴(kuò)展 中描述的一樣做盅。你還可以使用擴(kuò)展給泛型類型有條件的添加功能缤削,就像 具有泛型 Where 子句的擴(kuò)展 中描述的一樣。

注意

對一個(gè)現(xiàn)有的類型吹榴,如果你定義了一個(gè)擴(kuò)展來添加新的功能亭敢,那么這個(gè)類型的所有實(shí)例都可以使用這個(gè)新功能,包括那些在擴(kuò)展定義之前就存在的實(shí)例图筹。

計(jì)算型屬性

擴(kuò)展可以給現(xiàn)有類型添加計(jì)算型實(shí)例屬性和計(jì)算型類屬性帅刀。這個(gè)例子給 Swift 內(nèi)建的 Double 類型添加了五個(gè)計(jì)算型實(shí)例屬性,從而提供與距離單位相關(guān)工作的基本支持:

extension Double {
    var km: Double { return self * 1_000.0 }
    var m: Double { return self }
    var cm: Double { return self / 100.0 }
    var mm: Double { return self / 1_000.0 }
    var ft: Double { return self / 3.28084 }
}
let oneInch = 25.4.mm
print("One inch is \(oneInch) meters")
// 打印“One inch is 0.0254 meters”
let threeFeet = 3.ft
print("Three feet is \(threeFeet) meters")
// 打印“Three feet is 0.914399970739201 meters”

這些計(jì)算型屬性表示的含義是把一個(gè) Double 值看作是某單位下的長度值远剩。即使它們被實(shí)現(xiàn)為計(jì)算型屬性扣溺,但這些屬性的名字仍可緊接一個(gè)浮點(diǎn)型字面值,從而通過點(diǎn)語法來使用瓜晤,并以此實(shí)現(xiàn)距離轉(zhuǎn)換锥余。

在上述例子中,Double 類型的 1.0 代表的是“一米”痢掠。這就是為什么計(jì)算型屬性 m 返回的是 self——表達(dá)式 1.m 被認(rèn)為是計(jì)算一個(gè) Double 類型的 1.0驱犹。

其它單位則需要一些單位換算。一千米等于 1,000 米足画,所以計(jì)算型屬性 km 要把值乘以 1_000.00 來實(shí)現(xiàn)千米到米的單位換算着绷。類似地,一米有 3.28084 英尺锌云,所以計(jì)算型屬性 ft 要把對應(yīng)的 Double 值除以 3.28084荠医,來實(shí)現(xiàn)英尺到米的單位換算。

這些屬性都是只讀的計(jì)算型屬性桑涎,所以為了簡便彬向,它們的表達(dá)式里面都不包含 get 關(guān)鍵字。它們使用 Double 作為返回值類型攻冷,并可用于所有接受 Double 類型的數(shù)學(xué)計(jì)算中:

let aMarathon = 42.km + 195.m
print("A marathon is \(aMarathon) meters long")
// 打印“A marathon is 42195.0 meters long”

注意 擴(kuò)展可以添加新的計(jì)算屬性娃胆,但是它們不能添加存儲(chǔ)屬性,或向現(xiàn)有的屬性添加屬性觀察者等曼。

構(gòu)造器

擴(kuò)展可以給現(xiàn)有的類型添加新的構(gòu)造器里烦。它使你可以把自定義類型作為參數(shù)來供其他類型的構(gòu)造器使用,或者在類型的原始實(shí)現(xiàn)上添加額外的構(gòu)造選項(xiàng)禁谦。

擴(kuò)展可以給一個(gè)類添加新的便利構(gòu)造器胁黑,但是它們不能給類添加新的指定構(gòu)造器或者析構(gòu)器。指定構(gòu)造器和析構(gòu)器必須始終由類的原始實(shí)現(xiàn)提供州泊。

如果你使用擴(kuò)展給一個(gè)值類型添加構(gòu)造器丧蘸,而這個(gè)值類型已經(jīng)為所有存儲(chǔ)屬性提供默認(rèn)值,且沒有定義任何自定義構(gòu)造器遥皂,那么你可以在該值類型擴(kuò)展的構(gòu)造器中使用默認(rèn)構(gòu)造器和成員構(gòu)造器力喷。如果你已經(jīng)將構(gòu)造器寫在值類型的原始實(shí)現(xiàn)中刽漂,則不適用于這種情況,如同 值類型的構(gòu)造器代理 中所描述的那樣弟孟。

如果你使用擴(kuò)展給另一個(gè)模塊中定義的結(jié)構(gòu)體添加構(gòu)造器贝咙,那么新的構(gòu)造器直到定義模塊中使用一個(gè)構(gòu)造器之前,不能訪問 self拂募。

在下面的例子中庭猩,自定義了一個(gè)的 Rect 結(jié)構(gòu)體用來表示一個(gè)幾何矩形。這個(gè)例子中還定義了兩個(gè)給予支持的結(jié)構(gòu)體 SizePoint没讲,它們都把屬性的默認(rèn)值設(shè)置為 0.0

struct Size {
    var width = 0.0, height = 0.0
}
struct Point {
    var x = 0.0, y = 0.0
}
struct Rect {
    var origin = Point()
    var size = Size()
}

因?yàn)?Rect 結(jié)構(gòu)體給所有的屬性都提供了默認(rèn)值,所以它自動(dòng)獲得了一個(gè)默認(rèn)構(gòu)造器和一個(gè)成員構(gòu)造器礁苗,就像 默認(rèn)構(gòu)造器 中描述的一樣爬凑。這些構(gòu)造器可以用來創(chuàng)建新的 Rect 實(shí)例:

let defaultRect = Rect()
let memberwiseRect = Rect(origin: Point(x: 2.0, y: 2.0),
   size: Size(width: 5.0, height: 5.0))

你可以通過擴(kuò)展 Rect 結(jié)構(gòu)體來提供一個(gè)允許指定 point 和 size 的構(gòu)造器:

extension Rect {
    init(center: Point, size: Size) {
        let originX = center.x - (size.width / 2)
        let originY = center.y - (size.height / 2)
        self.init(origin: Point(x: originX, y: originY), size: size)
    }
}

這個(gè)新的構(gòu)造器首先根據(jù)提供的 centersize 計(jì)算一個(gè)適當(dāng)?shù)脑c(diǎn)。然后這個(gè)構(gòu)造器調(diào)用結(jié)構(gòu)體自帶的成員構(gòu)造器 init(origin:size:)试伙,它會(huì)將新的 origin 和 size 值儲(chǔ)存在適當(dāng)?shù)膶傩灾校?/p>

let centerRect = Rect(center: Point(x: 4.0, y: 4.0), size: Size(width: 3.0, height: 3.0))
// centerRect 的 origin 是 (2.5, 2.5) 并且它的 size 是 (3.0, 3.0)

注意

如果你通過擴(kuò)展提供一個(gè)新的構(gòu)造器嘁信,你有責(zé)任確保每個(gè)通過該構(gòu)造器創(chuàng)建的實(shí)例都是初始化完整的。

方法

擴(kuò)展可以給現(xiàn)有類型添加新的實(shí)例方法和類方法疏叨。在下面的例子中潘靖,給 Int 類型添加了一個(gè)新的實(shí)例方法叫做 repetitions

extension Int {
    func repetitions(task: () -> Void) {
        for _ in 0..<self {
            task()
        }
    }
}

repetitions(task:) 方法僅接收一個(gè) () -> Void 類型的參數(shù),它表示一個(gè)沒有參數(shù)沒有返回值的方法蚤蔓。

定義了這個(gè)擴(kuò)展之后卦溢,你可以對任意整形數(shù)值調(diào)用 repetitions(task:) 方法,來執(zhí)行對應(yīng)次數(shù)的任務(wù):

3.repetitions {
    print("Hello!")
}
// Hello!
// Hello!
// Hello!

可變實(shí)例方法

通過擴(kuò)展添加的實(shí)例方法同樣也可以修改(或 mutating(改變))實(shí)例本身秀又。結(jié)構(gòu)體和枚舉的方法单寂,若是可以修改 self 或者它自己的屬性,則必須將這個(gè)實(shí)例方法標(biāo)記為 mutating吐辙,就像是改變了方法的原始實(shí)現(xiàn)宣决。

在下面的例子中,對 Swift 的 Int 類型添加了一個(gè)新的 mutating 方法昏苏,叫做 square尊沸,它將原始值求平方:

extension Int {
    mutating func square() {
        self = self * self
    }
}
var someInt = 3
someInt.square()
// someInt 現(xiàn)在是 9

下標(biāo)

擴(kuò)展可以給現(xiàn)有的類型添加新的下標(biāo)。下面的例子中贤惯,對 Swift 的 Int 類型添加了一個(gè)整數(shù)類型的下標(biāo)洼专。下標(biāo) [n] 從數(shù)字右側(cè)開始,返回小數(shù)點(diǎn)后的第 n 位:

  • 123456789[0] 返回 9

  • 123456789[1] 返回 8

……以此類推:

extension Int {
    subscript(digitIndex: Int) -> Int {
        var decimalBase = 1
        for _ in 0..<digitIndex {
            decimalBase *= 10
        }
        return (self / decimalBase) % 10
    }
}
746381295[0]
// 返回 5
746381295[1]
// 返回 9
746381295[2]
// 返回 2
746381295[8]
// 返回 7

如果操作的 Int 值沒有足夠的位數(shù)滿足所請求的下標(biāo)孵构,那么下標(biāo)的現(xiàn)實(shí)將返回 0壶熏,將好像在數(shù)字的左邊補(bǔ)上了 0:

746381295[9]
// 返回 0,就好像你進(jìn)行了這個(gè)請求:
0746381295[9]

嵌套類型

擴(kuò)展可以給現(xiàn)有的類浦译,結(jié)構(gòu)體棒假,還有枚舉添加新的嵌套類型:

extension Int {
    enum Kind {
        case negative, zero, positive
    }
    var kind: Kind {
        switch self {
        case 0:
            return .zero
        case let x where x > 0:
            return .positive
        default:
            return .negative
        }
    }
}

這個(gè)例子給 Int 添加了一個(gè)新的嵌套枚舉溯职。這個(gè)枚舉叫做 Kind,表示特定整數(shù)所代表的數(shù)字類型帽哑。具體來說谜酒,它表示數(shù)字是負(fù)的、零的還是正的妻枕。

這個(gè)例子同樣給 Int 添加了一個(gè)新的計(jì)算型實(shí)例屬性围段,叫做 kind,它返回被操作整數(shù)所對應(yīng)的 Kind 枚舉 case 分支氛赐。

現(xiàn)在籍琳,任意 Int 的值都可以使用這個(gè)嵌套類型:

func printIntegerKinds(_ numbers: [Int]) {
    for number in numbers {
        switch number.kind {
        case .negative:
            print("- ", terminator: "")
        case .zero:
            print("0 ", terminator: "")
        case .positive:
            print("+ ", terminator: "")
        }
    }
    print("")
}
printIntegerKinds([3, 19, -27, 0, -6, 0, 7])
// 打印“+ + - 0 - 0 + ”

方法 printIntegerKinds(_:),使用一個(gè) Int 類型的數(shù)組作為輸入愕掏,然后依次迭代這些值度秘。對于數(shù)組中的每一個(gè)整數(shù),方法會(huì)檢查它的 kind 計(jì)算型屬性饵撑,然后打印適當(dāng)?shù)拿枋觥?/p>

注意 number.kind 已經(jīng)被認(rèn)為是 Int.Kind 類型剑梳。所以,在 switch 語句中所有的 Int.Kind case 分支可以被縮寫滑潘,就像使用 .negative 替代 Int.Kind.negative.垢乙。


繼續(xù)閱讀 Swift - 協(xié)議

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市语卤,隨后出現(xiàn)的幾起案子追逮,更是在濱河造成了極大的恐慌,老刑警劉巖粹舵,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件羊壹,死亡現(xiàn)場離奇詭異,居然都是意外死亡齐婴,警方通過查閱死者的電腦和手機(jī)油猫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來柠偶,“玉大人情妖,你說我怎么就攤上這事∮盏#” “怎么了毡证?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長蔫仙。 經(jīng)常有香客問我料睛,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任恤煞,我火速辦了婚禮屎勘,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘居扒。我一直安慰自己概漱,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布喜喂。 她就那樣靜靜地躺著瓤摧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪玉吁。 梳的紋絲不亂的頭發(fā)上照弥,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天,我揣著相機(jī)與錄音进副,去河邊找鬼这揣。 笑死,一個(gè)胖子當(dāng)著我的面吹牛敢会,可吹牛的內(nèi)容都是我干的曾沈。 我是一名探鬼主播这嚣,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼鸥昏,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了姐帚?” 一聲冷哼從身側(cè)響起吏垮,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎罐旗,沒想到半個(gè)月后膳汪,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡九秀,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年遗嗽,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鼓蜒。...
    茶點(diǎn)故事閱讀 40,137評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡痹换,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出都弹,到底是詐尸還是另有隱情娇豫,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布畅厢,位于F島的核電站冯痢,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜浦楣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一袖肥、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧椒振,春花似錦昭伸、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至夹供,卻和暖如春灵份,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背哮洽。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工填渠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鸟辅。 一個(gè)月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓氛什,卻偏偏與公主長得像,于是被迫代替她去往敵國和親匪凉。 傳聞我的和親對象是個(gè)殘疾皇子枪眉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 擴(kuò)展 就是為一個(gè)已有的類、結(jié)構(gòu)體再层、枚舉類型或者協(xié)議類型添加新功能贸铜。這包括在沒有權(quán)限獲取原始源代碼的情況下擴(kuò)展類型的...
    CDLOG閱讀 165評論 0 0
  • (轉(zhuǎn)載自http://letsswift.com/2014/06/extensions/) 擴(kuò)展就是向一個(gè)已有的類...
    StrongX閱讀 1,610評論 0 11
  • 中文文檔 擴(kuò)展 就是為一個(gè)已有的類、結(jié)構(gòu)體聂受、枚舉類型或者協(xié)議類型添加新功能蒿秦。這包括在沒有權(quán)限獲取原始源代碼的情況下...
    伯wen閱讀 242評論 0 1
  • 擴(kuò)展可以給一個(gè)現(xiàn)有的類,結(jié)構(gòu)體蛋济,枚舉棍鳖,還有協(xié)議添加新的功能。 擁有不需要訪問被擴(kuò)展類型源代碼就能完成擴(kuò)展的能力(即...
    DevXue閱讀 235評論 0 0
  • 擴(kuò)展就是向一個(gè)已有的類碗旅、結(jié)構(gòu)體或枚舉類型添加新功能渡处。擴(kuò)展可以對一個(gè)類型添加新的功能,但是不能重寫已有的功能扛芽。Swi...
    零度_不結(jié)冰閱讀 287評論 0 0