[翻]如何将您的APP从Objective C转移到Swift?

1,265 阅读6分钟

自从苹果在2014年发布了自己的Swift编程语言之后,很多开发人员都利用这个机会来测试它的特性,看看它是如何与Objective-C进行比较的。 一般结论似乎是: 在某些情况下,Swift比Objective-C更可取。 ###Swift的优点是什么?

  1. Swift允许你编写较少的代码。
  2. Swift是强类型的,这意味着更少的处理类型不正确导致崩溃。
  3. Swift看起来类似于其他流行的编程语言,类似于英语。
  4. 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?您有两个选项从可选容器获取包装的值。第一个是可选链接 - 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中的整个现有代码库。