WWDC之Swift Interoperability in Depth

前言

Session,主要为进一步讲解Cocoa框架中Objective-CSwift转换以及两种如何更好共同协作;

  • Working With Cocoa: 使用Swift如何与Cocoa更好协作开发
  • Bridging Core Cocoa Types: Cocoa核心类型到Swift的桥接
  • CF Interoperability: SwiftCore FoundationC语言系列API的协作

内容

Working With Cocoa

  • 属性的转化

    // Objective-C
    @property (nonatomic, strong) NSData *data;
    
    // Swift
    var data:NSData!
    

    Objective-C中普通的属性会转换成带末尾带!的Swift属性:

    1. 在用Objective-C实现的Cocoa框架中的类的实例直接可以被设为nil,而Swift中不存在可以被设为nil的类实例对象,只有成为Optional类型的类实例,才能显示设为nil;
    2. !既表示其属于Optional类型,又将其进行强制解包操作,使得外部允许直接访问真正的值.(在Swift中使用时需要避免当该值为nil情况下访问,会造成Crash)
  • 方法的转化

    // Objective-C
    - (void)fetchDataWithToken:(NSString *)token Password:(NSString *)password;
                                              // 参数标签          // 内部参数
    // Swift
    func fetchDataWithToken(token: String, password:String) -> Void {。。。}
    
    fetchDataWithToken(token:"...",password:"...") // executing
    

Objective-C方法的书写中需要显示写明参数标签(即外部参数),和内部参数;而Swift中默认参数标签与内部参数名默认一致(除首个参数外),也允许额外设置参数标签甚至使用_进行省略.

  • Block 与 Closure

    // Objective-C
    (void)(^Success)()  
    // Swift
    success:(()-> Void)!
    

    与属性的转换相似,其末尾也带!,表示该闭包可以为设为nil;
    除此自外,在Swift的方法中的闭包若最为最后一个参数传入,则可以使用trailing closure语法,让方法可读性更高.

    func addOperationWithA(a:String, operation:() -> Void) {
        operation()
    }
    addOperationWithA("a") { () -> Void in
        // code
    }
    
  • 初始化方法的转化

    // Objective-C
    - (instancetype)initWithBytesNoCopy:(void *)bytes length:(NSUInteger)length;
    
    // Swift
    init(bytesNoCopy bytes: UnsafeMutablePointer, length: Int)
    

Objective-Cinit方法转换时会将其紧跟With后的字符串首字母小写,作为Swift初始化方法的第一个参数变量的参数标签

  • NSError的转换
    常见的参数NSError**类型转化 NSErrorPointer指针类型(若要使用该指针上的内存,可以访问memory属性),若对方法中error不关心,可直接设nil

  • Enum的转化
    转换后移除了充当命名空间的枚举类型名称前缀,并且用点语法表示枚举变量,依靠类型推断可以允许.前的枚举类型省略.

  • id 与 AnyObject

  1. 为了对应Objective-Cid类型,Swift使用AnyObject(内容为空的协议)表示可能的任何类型,依赖运行时进行具体类型的确定;
  2. id类型变量在调用指定selector前通常使用respondsToSelector进行判断,而转换后成为了AnyObject?,AnyObject的任何方法调用都是Optional, 意味着若selector不存在,整个方法调用为nil,其内部自动进行了respondsToSelector的调用判断.
var object: AnyObject = NSData()
object.removeLastObject?() // nil
       // NSArray Method
  1. AnyObject类型对象进行类型转换时使用as关键字,如果进行downcast(强制转换),使用as?进行指定类型转换,并且检查转换成功后进行使用.
  • 协议的转化
    对应属性设置id delegate,转换后利用协议类型直接写为var delegate: XXXDelegate

Bridging Core Cocoa Types

下图为Cocoa框架中重要类型的对应转换:

WWDC之Swift Interoperability in Depth_第1张图片
Cocoa核心类型Bridge.png

举例: NSArray的Swift转化

  1. Objective-CNSArray都会转换成原生Array,具体为AnyObject[]!(备注:后面的!表示该数组为Optional类型数组,且可以设为nil)
  2. 任何某一特定类型的Array都可以直接赋值给AnyObject[]类型的数组,而在使用AnyObject[]时可以用as T[]进行特定类型数组的强制转换后进一步使用.

Subclassing Objective-C Classes

  • override的使用
    继承父类后重写方法的关键字,提供了更加安全的代码错误检测
    利用override进行属性的重写,setget都在同一处内重写

    override var description: String {
       get { return "It's override description" }
       set {...}
    }
    
  • @objc关键字的使用

    1. 由于Swift所独有的特性如元组,泛型,枚举和结构体对象等不能在Objective-C中找到对应的转换 可以在方法或属性前加``@objc关键字,可以用来检测能否以Objective-C`形式表现否则出编译错误.
    2. 控制Objective-C中的命名
    var enabled: Bool {
          @objc(isEnabled) get {...}  // getter is named "isEnabled" in `Objective-C`
          set {...}
    }
          @objc(XYZMyDocument) class MyDocument:UIDocument {   // class is named `XYZMyDocument` in `Objective-C`
        // ...
    }
    

CF Interoperability

SwiftCore Foundation的大部分API都进行了隐式桥接,而不需在自己的代码中写过多与其交互的桥接代码和内存管理,Swift都自动帮助管理其CF对象.
对于需要显示进行桥接的CF APIs,Swift使用Unmanaged结构体对象表示该对象的内存需要手动管理,主要有以下方法

takeRetainedValue() -> Instance  // +1 returns
takeUnretainedValue() -> Instance // +0 returns

总结

通过Objective-CSwift之间的相互转化的最终目的是让其更加高效安全地构建App,让Objective-C开发者能更好地理解和掌握Swift开发.

你可能感兴趣的:(WWDC之Swift Interoperability in Depth)