前言
本Session
,主要为进一步讲解Cocoa框架中Objective-C
向Swift
转换以及两种如何更好共同协作;
-
Working With Cocoa
: 使用Swift
如何与Cocoa
更好协作开发 -
Bridging Core Cocoa Types
:Cocoa
核心类型到Swift
的桥接 -
CF Interoperability
:Swift
与Core Foundation
等C
语言系列API的协作
内容
Working With Cocoa
-
属性的转化
// Objective-C @property (nonatomic, strong) NSData *data; // Swift var data:NSData!
在
Objective-C
中普通的属性会转换成带末尾带!
的Swift属性:- 在用
Objective-C
实现的Cocoa
框架中的类的实例直接可以被设为nil
,而Swift
中不存在可以被设为nil
的类实例对象,只有成为Optional
类型的类实例,才能显示设为nil
; -
!
既表示其属于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<Void>, length: Int)
Objective-C
的init
方法转换时会将其紧跟With
后的字符串首字母小写,作为Swift
初始化方法的第一个参数变量的参数标签
NSError的转换
常见的参数NSError**
类型转化NSErrorPointer
指针类型(若要使用该指针上的内存,可以访问memory
属性),若对方法中error
不关心,可直接设nil
Enum
的转化
转换后移除了充当命名空间的枚举类型名称前缀,并且用点语法表示枚举变量,依靠类型推断可以允许.
前的枚举类型省略.id 与 AnyObject
- 为了对应
Objective-C
的id
类型,Swift
使用AnyObject
(内容为空的协议)表示可能的任何类型,依赖运行时进行具体类型的确定; - 对
id
类型变量在调用指定selector
前通常使用respondsToSelector
进行判断,而转换后成为了AnyObject?
,AnyObject
的任何方法调用都是Optional
, 意味着若selector
不存在,整个方法调用为nil
,其内部自动进行了respondsToSelector
的调用判断.
var object: AnyObject = NSData()
object.removeLastObject?() // nil
// NSArray Method
- 将
AnyObject
类型对象进行类型转换时使用as
关键字,如果进行downcast
(强制转换),使用as?
进行指定类型转换,并且检查转换成功后进行使用.
- 协议的转化
对应属性设置id <XXXDelegate> delegate
,转换后利用协议类型直接写为var delegate: XXXDelegate
Bridging Core Cocoa Types
下图为Cocoa
框架中重要类型的对应转换:
举例: NSArray的Swift转化
-
Objective-C
中NSArray
都会转换成原生Array
,具体为AnyObject[]!
(备注:后面的!
表示该数组为Optional
类型数组,且可以设为nil
) - 任何某一特定类型的
Array
都可以直接赋值给AnyObject[]
类型的数组,而在使用AnyObject[]
时可以用as T[]
进行特定类型数组的强制转换后进一步使用.
Subclassing Objective-C Classes
-
override
的使用
继承父类后重写方法的关键字,提供了更加安全的代码错误检测
利用override
进行属性的重写,set
和get
都在同一处内重写override var description: String { get { return "It's override description" } set {...} }
-
@objc
关键字的使用- 由于
Swift
所独有的特性如元组,泛型,枚举和结构体对象等不能在Objective-C
中找到对应的转换 可以在方法或属性前加``@objc关键字,可以用来检测能否以
Objective-C`形式表现否则出编译错误. - 控制
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
Swift
对Core Foundation
的大部分API
都进行了隐式桥接,而不需在自己的代码中写过多与其交互的桥接代码和内存管理,Swift
都自动帮助管理其CF
对象.
对于需要显示进行桥接的CF APIs
,Swift
使用Unmanaged<Instance>
结构体对象表示该对象的内存需要手动管理,主要有以下方法
takeRetainedValue() -> Instance // +1 returns
takeUnretainedValue() -> Instance // +0 returns
总结
通过Objective-C
与Swift
之间的相互转化的最终目的是让其更加高效安全地构建App
,让Objective-C
开发者能更好地理解和掌握Swift
开发.