譯者注:這篇文章是Reflectable enums in Swift 3的中文翻譯
如果你正在用siwft編寫(xiě)程序并且需要一個(gè)數(shù)據(jù)模型來(lái)表示聯(lián)系人疹鳄,比如說(shuō)一個(gè)聯(lián)系人可能是來(lái)自你的通訊錄或者是FaceBoook的朋友瘪弓。你可能像下面一樣初始化一個(gè)結(jié)構(gòu)體的Contact:
這種方法非常簡(jiǎn)單方便但是有一個(gè)顯著地缺點(diǎn)腺怯。注意到結(jié)構(gòu)體Contact有兩個(gè)可選的屬性:phoneNUmber和username.這兩個(gè)屬性是根據(jù)不同的聯(lián)系人類(lèi)型有條件的分配一個(gè)值呛占。
除非開(kāi)發(fā)者碰巧看到屬性的注釋,否則的話每個(gè)屬性都不是非空的規(guī)則是不明顯的疹味。這樣的設(shè)計(jì)很容易使開(kāi)發(fā)者犯錯(cuò)糙捺。如果我們不能意外的訪問(wèn)到了一個(gè)屬性空洪灯,顯然會(huì)更好婴渡,因?yàn)槟阃洐z查聯(lián)系人的類(lèi)型凯亮。
一種更普遍的解決方式是創(chuàng)建像下面的枚舉:
聯(lián)系人枚舉類(lèi)型有兩種柠并,每一種類(lèi)型都有自己關(guān)聯(lián)的struct類(lèi)型富拗。當(dāng)我們創(chuàng)建這樣的枚舉的時(shí)候不可能出現(xiàn)偶然訪問(wèn)錯(cuò)誤的屬性啃沪,比如說(shuō)AddressBook 聯(lián)系人的username屬性或者是FaceBook聯(lián)系人的phoneNumber屬性,因?yàn)檫@時(shí)候它們對(duì)于每一種聯(lián)系人已經(jīng)在定義是獨(dú)立分開(kāi)缰雇。這些結(jié)構(gòu)體包含了正確的values和他們關(guān)聯(lián)的聯(lián)系人類(lèi)型械哟。哈哈殿雪,現(xiàn)在不在需要可選的屬性了暇咆!
然后這種方式還有一種缺點(diǎn)。大多數(shù)場(chǎng)景下,已一種統(tǒng)一的方式訪問(wèn)聯(lián)系人枚舉中共同的屬性是便捷的爸业,例如像下面這樣:
為了解決這種問(wèn)題其骄,你可以在Contact的枚舉中創(chuàng)建一個(gè)計(jì)算屬性,計(jì)算屬性根據(jù)枚舉關(guān)聯(lián)的struct類(lèi)型在switch的block中讀取扯旷。
這個(gè)簡(jiǎn)單的例子入上面的代碼看起來(lái)似乎沒(méi)有太大的問(wèn)題年栓。但是真正開(kāi)發(fā)程序的時(shí)候,數(shù)據(jù)model可能會(huì)有許多許多的屬性薄霜,這時(shí)候各種各樣的model我們都需要去用這種代碼去處理某抓,這樣這種代碼會(huì)很快變得一個(gè)繁瑣的編寫(xiě)并且難以維護(hù)。顯然惰瓜,無(wú)聊冗余的代碼是bugs的源頭。我們必須有一種更好的方式去解決。曲尸。。
這種問(wèn)題可以通過(guò)Swift中的反射API去解決昆箕。反射是Swift一種內(nèi)省代碼的方式,換句話誰(shuí)就是在運(yùn)行時(shí)可以可以自己纤泵。和其他平臺(tái)比較,比如.Net涉馅,Java捻浦,swift反射只提供了一些非持觳樱基本的功能,但在這里已經(jīng)足夠我們使用了甸祭。
讓我們定義一個(gè)RefrectableEnum 協(xié)議:
這個(gè)協(xié)議唯一的方法在協(xié)議的擴(kuò)展中提供了默認(rèn)實(shí)現(xiàn)凡怎。它使用反射在enum內(nèi)部關(guān)聯(lián)struct并且找到其中的屬性。該協(xié)議能夠減輕我們之前代碼的編碼負(fù)擔(dān)凝赛,例如:
重要的是添加到Contact的屬性具有和關(guān)聯(lián)的struct枚舉cases完全相同的屬性赚楚。使用Swift的#function 意味著調(diào)用value(of :)的屬性的名稱作為propertyName參數(shù)傳入(譯者注:實(shí)際上上述例子中你也完全可以使用屬性對(duì)應(yīng)的名稱字符串,這里看起來(lái)高大上而已...)烤宙。
上面代碼使用了向下強(qiáng)轉(zhuǎn)躺枕,因?yàn)椴糠执a是缺少安全性的。它依賴于命名的約定膳帕。編譯器不能保證在運(yùn)行時(shí)編譯成功。value(of:)的實(shí)現(xiàn)中使用了可選類(lèi)型的強(qiáng)制解包,如果初始化不能匹配上可能會(huì)使代碼crash。這里我的目的是遇到問(wèn)題就crash(比如說(shuō)關(guān)聯(lián)的struct的屬性沒(méi)有一個(gè)確定的name和type)狱掂,但這是必要的惦蚊,因?yàn)榫幾g器不能警告這種代碼提出修改建議兆沙。如果你想編譯器能夠容錯(cuò)處理憎妙,那么我的解決方案可能不太適合你褥符。
上面的代碼僅僅是一種粗濾的草稿,他可能不太符合你的應(yīng)用,但為未來(lái)的優(yōu)化提供了一個(gè)很好的開(kāi)始粗截。上面的代碼包含ReflectableEnum你可以在下面的鏈接中找到:
https://github.com/ijoshsmith/reflectable-enum
愉快的使用Swift吧婿屹!