前言
本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框架中重要类型的对应转换:
Cocoa核心类型Bridge.png
举例: 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开发.










网友评论