自從蘋果在2014年發(fā)布了自己的Swift編程語言之后获印,很多開發(fā)人員都利用這個機(jī)會來測試它的特性,看看它是如何與Objective-C進(jìn)行比較的胆建。 一般結(jié)論似乎是: 在某些情況下烤低,Swift比Objective-C更可取。
Swift的優(yōu)點是什么笆载?
- Swift允許你編寫較少的代碼扑馁。
- Swift是強(qiáng)類型的,這意味著更少的處理類型不正確導(dǎo)致崩潰凉驻。
- Swift看起來類似于其他流行的編程語言腻要,類似于英語。
- Swift比它的前任Objective-C快涝登。
這些功能使Swift非常誘人雄家。 讓我們考慮如果你決定從Objective-C切換到Swift你需要做什么,并確定是否應(yīng)該轉(zhuǎn)移到Swift上胀滚。
值得一提的是趟济,Swift與Objective-C完全兼容。 Apple提供了一個混合和匹配功能咽笼,允許開發(fā)人員在同一個項目中使用這兩種語言顷编。 這意味著您可以向現(xiàn)有的代碼庫添加新的Swift功能。 同時剑刑,盡管Swift和Objective-C使用相同的API媳纬,但在從Objective-C遷移到Swift時,您必須記住一些差異施掏。
Optional 類型
在Objective-C中钮惠,你可以調(diào)用nil對象的方法(更準(zhǔn)確地說,你可以發(fā)送消息到nil對象)七芭,這些方法返回一個空值素挽。為了防止在出現(xiàn)意外的nil值時出現(xiàn)未定義的行為,您需要在需要時執(zhí)行nil檢查抖苦。 Swift引入了可選值的概念∶姿溃可選類型聲明為類似枚舉:
public enum可選<Wrapped>:_Reflectable锌历,NilLiteralConvertible
從編程上來說,它是一種類型峦筒,可以代表Wrapped類型的值或不存在的值究西。 Swift提供了使類型可選的語法糖,所以不需要聲明Optional <String>物喷,你可以只寫String卤材?您有兩個選項從可選容器獲取包裝的值遮斥。第一個是可選鏈接 - if-let條件語句僅當(dāng)它存在時才接收該值。如果您完全確定可選變量是非零扇丛,您可以使用強(qiáng)制解包术吗。這提供了存儲的值,如果存在帆精,沒有條件较屿,但是當(dāng)你犯了一個錯誤,并且可選實例為空時崩潰卓练。
除了常規(guī)可選項隘蝎,還有隱含的展開可選項,聲明為String襟企!讓我們來看看你可以用不同的方式聲明可選性嘱么。
class ExampleClass {
var nonOptionalString: String
var unwrappedOptionalString: String!
var optionalString: String?
init(string: String) {
nonOptionalString = string
}
}
nonOptionalString永遠(yuǎn)不能為nil。 此屬性應(yīng)在對象初始化期間填充顽悼。 提供強(qiáng)制解包的nil對象將導(dǎo)致崩潰曼振。
unwrappedOptionalString可以是nil,但如果您嘗試訪問nil對象表蝙,您的程序?qū)⒈罎ⅰ?br>
optionalString可以為nil拴测,應(yīng)被視為常規(guī)可選變量。
當(dāng)編寫Objective-C代碼時府蛇,可以使用_Nullable和_Nonnull類型注釋來標(biāo)記變量集索。 上面的Swift示例的Objective-C等價類似于:
@interface ExampleClass: NSObject
@property (nonatomic, strong) NSString * _Nonnull nonOptionalString;
@property (nonatomic, strong) NSString *unwrappedOptionalString;
@property (nonatomic, strong) NSString * _Nullable optionalString;
- (instancetype)initWithString:(nonnull NSString *string);
@end
錯誤處理
當(dāng)在Objective-C中拋出和處理錯誤時,方法的最后一個參數(shù)是對NSError變量的引用汇跨。 如果此方法執(zhí)行導(dǎo)致不可接受的行為务荆,則應(yīng)創(chuàng)建一個NSError實例并將其寫入傳遞的變量。 在調(diào)用可能產(chǎn)生錯誤的方法后穷遂,您應(yīng)該檢查錯誤參數(shù)函匕,以確保它是非零。
- (nonnull NSString *)exampleMethod:(nonnull NSString *)param error:(NSError **)error {
if (!param.length) {
*error = [NSError errorWithDomain:[NSBundle mainBundle].bundleIdentifier
code:-101
userInfo:nil];
return nil;
} // do work}
Objective-C還提供了一個具有傳統(tǒng)try-catch-finally語法的異常機(jī)制蚪黑,但是Apple強(qiáng)烈建議僅將其用于開發(fā)目的盅惜。
Swift要求您標(biāo)記使用throws關(guān)鍵字生成錯誤的方法。 如果該方法接受的最后一個參數(shù)是指向Objective-C接口中的NSError實例的指針忌穿,那么它將從Objective-C翻譯為Swift作為throwing方法抒寂,上面方法的聲明將轉(zhuǎn)換為 :
func exampleMethod(param:String)throws - > String
Objective-C允許你省略錯誤處理的錯誤返回方法,但在Swift你必須明確處理錯誤掠剑。 被拋出的對象應(yīng)該是Swift ErrorType的后代屈芜。
枚舉
Objective-C提供C風(fēng)格的枚舉,它只限于基本類型。 即使您需要將整數(shù)枚舉值映射到相應(yīng)的字符串以顯示給用戶或發(fā)送到后端井佑,也必須創(chuàng)建數(shù)組或字典 - 或使用switch語句属铁。 但Swift提供了全新的枚舉與更多的選擇。 Swift中的枚舉可以按照與Objective-C中相同的方式使用:
enum ExampleEnum {
case ExOne, ExTwo, ExThree
}
Swift枚舉可以存儲關(guān)聯(lián)的值躬翁。 每個枚舉大小寫都可以包含一組預(yù)定義的字段焦蘑。
enum AnotherExampleEnum {
case ExOne(String, Int)
case ExTwo(Int)
}
Swift枚舉可以存儲原始值并遞歸。
Swift還有各種強(qiáng)大的功能姆另,它與Objective-C區(qū)別開來喇肋。 這些功能包括泛型,嚴(yán)格類型系統(tǒng)迹辐,類型推斷蝶防,元組和嵌套類型。 簡而言之明吩,從Objective-C遷移到Swift不是一件小事间学。 這里有一些提示,使過程更容易印荔。
從Objective-C遷移到Swift的提示
首先低葫,您應(yīng)該創(chuàng)建一個與相應(yīng)頭和實現(xiàn)文件名稱相同的.swift文件。 如果你需要從新的Swift文件中訪問Objective-C類仍律,你必須為它們添加一個import指令到橋接頭文件嘿悬。 如果你需要從Objective-C代碼訪問新的Swift類,你必須從Objective-C類繼承它; 否則將無法訪問水泉。 然后你必須手動重寫代碼善涨,采用Swift的最佳實踐并使類向下兼容。
如果你的Google“從Objective-C遷移到Swift”草则,你會發(fā)現(xiàn)幾個鏈接钢拧,建議自動翻譯代碼的工具。 有免費和付費解決方案炕横。 然而源内,即使付費的解決方案在它們的功能上極其有限。
為了看看這些工具是如何在Objective-C中創(chuàng)建一個非撤莸睿基本的游戲膜钓,然后嘗試使用幾個自動翻譯器將它翻譯成Swift。我們嘗試的第一個是基于Web的卿嘲,并允許您上傳一個完整的Xcode項目(但是只有當(dāng)項目小于10 MB)颂斜。結(jié)果真的沒有準(zhǔn)備好生產(chǎn) - 我們發(fā)現(xiàn)超過70個錯誤,即使我們的程序只有7個小類腔寡。這些錯誤包括不正確的可選焚鲜,不正確的類型推斷掌唾,留下最后一個參數(shù)為NSError放前,而不是切換到Swift錯誤處理與try-catch忿磅,從__weak不正確切換typeof(self)weakSelf = self;到[weak self]等。我們還測試了一個付費桌面應(yīng)用程序凭语,花費15美元葱她,但結(jié)果更糟。我們測試的桌面應(yīng)用程序無法導(dǎo)入整個項目似扔,因此我們不得不按文件復(fù)制和粘貼所有文件吨些。
行業(yè)中的大多數(shù)人都同意Swift正在成為iOS開發(fā)的主要語言,因此建議將它用于新項目炒辉。
方便地是豪墅,混合和匹配特性允許您使用Swift和Objective-C類作為同一項目的一部分,因此您不必花費時間重寫Swift中的整個現(xiàn)有代碼庫黔寇。