自從蘋果在2014年發布了自己的Swift編程語言之后,很多開發人員都利用這個機會來測試它的特性,看看它是如何與Objective-C進行比較的。 一般結論似乎是: 在某些情況下,Swift比Objective-C更可取。
Swift的優點是什么?
- Swift允許你編寫較少的代碼。
- Swift是強類型的,這意味著更少的處理類型不正確導致崩潰。
- Swift看起來類似于其他流行的編程語言,類似于英語。
- Swift比它的前任Objective-C快。
這些功能使Swift非常誘人。 讓我們考慮如果你決定從Objective-C切換到Swift你需要做什么,并確定是否應該轉移到Swift上。
值得一提的是,Swift與Objective-C完全兼容。 Apple提供了一個混合和匹配功能,允許開發人員在同一個項目中使用這兩種語言。 這意味著您可以向現有的代碼庫添加新的Swift功能。 同時,盡管Swift和Objective-C使用相同的API,但在從Objective-C遷移到Swift時,您必須記住一些差異。
Optional 類型
在Objective-C中,你可以調用nil對象的方法(更準確地說,你可以發送消息到nil對象),這些方法返回一個空值。為了防止在出現意外的nil值時出現未定義的行為,您需要在需要時執行nil檢查。 Swift引入了可選值的概念。可選類型聲明為類似枚舉:
public enum可選<Wrapped>:_Reflectable,NilLiteralConvertible
從編程上來說,它是一種類型,可以代表Wrapped類型的值或不存在的值。 Swift提供了使類型可選的語法糖,所以不需要聲明Optional <String>,你可以只寫String?您有兩個選項從可選容器獲取包裝的值。第一個是可選鏈接 - if-let條件語句僅當它存在時才接收該值。如果您完全確定可選變量是非零,您可以使用強制解包。這提供了存儲的值,如果存在,沒有條件,但是當你犯了一個錯誤,并且可選實例為空時崩潰。
除了常規可選項,還有隱含的展開可選項,聲明為String!讓我們來看看你可以用不同的方式聲明可選性。
class ExampleClass {
var nonOptionalString: String
var unwrappedOptionalString: String!
var optionalString: String?
init(string: String) {
nonOptionalString = string
}
}
nonOptionalString永遠不能為nil。 此屬性應在對象初始化期間填充。 提供強制解包的nil對象將導致崩潰。
unwrappedOptionalString可以是nil,但如果您嘗試訪問nil對象,您的程序將崩潰。
optionalString可以為nil,應被視為常規可選變量。
當編寫Objective-C代碼時,可以使用_Nullable和_Nonnull類型注釋來標記變量。 上面的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
錯誤處理
當在Objective-C中拋出和處理錯誤時,方法的最后一個參數是對NSError變量的引用。 如果此方法執行導致不可接受的行為,則應創建一個NSError實例并將其寫入傳遞的變量。 在調用可能產生錯誤的方法后,您應該檢查錯誤參數,以確保它是非零。
- (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還提供了一個具有傳統try-catch-finally語法的異常機制,但是Apple強烈建議僅將其用于開發目的。
Swift要求您標記使用throws關鍵字生成錯誤的方法。 如果該方法接受的最后一個參數是指向Objective-C接口中的NSError實例的指針,那么它將從Objective-C翻譯為Swift作為throwing方法,上面方法的聲明將轉換為 :
func exampleMethod(param:String)throws - > String
Objective-C允許你省略錯誤處理的錯誤返回方法,但在Swift你必須明確處理錯誤。 被拋出的對象應該是Swift ErrorType的后代。
枚舉
Objective-C提供C風格的枚舉,它只限于基本類型。 即使您需要將整數枚舉值映射到相應的字符串以顯示給用戶或發送到后端,也必須創建數組或字典 - 或使用switch語句。 但Swift提供了全新的枚舉與更多的選擇。 Swift中的枚舉可以按照與Objective-C中相同的方式使用:
enum ExampleEnum {
case ExOne, ExTwo, ExThree
}
Swift枚舉可以存儲關聯的值。 每個枚舉大小寫都可以包含一組預定義的字段。
enum AnotherExampleEnum {
case ExOne(String, Int)
case ExTwo(Int)
}
Swift枚舉可以存儲原始值并遞歸。
Swift還有各種強大的功能,它與Objective-C區別開來。 這些功能包括泛型,嚴格類型系統,類型推斷,元組和嵌套類型。 簡而言之,從Objective-C遷移到Swift不是一件小事。 這里有一些提示,使過程更容易。
從Objective-C遷移到Swift的提示
首先,您應該創建一個與相應頭和實現文件名稱相同的.swift文件。 如果你需要從新的Swift文件中訪問Objective-C類,你必須為它們添加一個import指令到橋接頭文件。 如果你需要從Objective-C代碼訪問新的Swift類,你必須從Objective-C類繼承它; 否則將無法訪問。 然后你必須手動重寫代碼,采用Swift的最佳實踐并使類向下兼容。
如果你的Google“從Objective-C遷移到Swift”,你會發現幾個鏈接,建議自動翻譯代碼的工具。 有免費和付費解決方案。 然而,即使付費的解決方案在它們的功能上極其有限。
為了看看這些工具是如何在Objective-C中創建一個非常基本的游戲,然后嘗試使用幾個自動翻譯器將它翻譯成Swift。我們嘗試的第一個是基于Web的,并允許您上傳一個完整的Xcode項目(但是只有當項目小于10 MB)。結果真的沒有準備好生產 - 我們發現超過70個錯誤,即使我們的程序只有7個小類。這些錯誤包括不正確的可選,不正確的類型推斷,留下最后一個參數為NSError,而不是切換到Swift錯誤處理與try-catch,從__weak不正確切換typeof(self)weakSelf = self;到[weak self]等。我們還測試了一個付費桌面應用程序,花費15美元,但結果更糟。我們測試的桌面應用程序無法導入整個項目,因此我們不得不按文件復制和粘貼所有文件。
行業中的大多數人都同意Swift正在成為iOS開發的主要語言,因此建議將它用于新項目。
方便地是,混合和匹配特性允許您使用Swift和Objective-C類作為同一項目的一部分,因此您不必花費時間重寫Swift中的整個現有代碼庫。