Associated Object的使用
熟悉OC的人估計(jì)都知道如何通過(guò)Category給已有的類添加屬性秩仆,那就是通過(guò)runtime的Associated Object這種方式舒裤。而在Swift中這種方法依然有效,對(duì)應(yīng)的API如下
C
id objc_getAssociatedObject(id object,
const void *key);
void objc_setAssociatedObject(id object,
const void *key,
id value,
objc_AssociationPolicy policy);
Swift
public func objc_getAssociatedObject(_ object: Any!,
_ key: UnsafeRawPointer!) -> Any!
public func objc_setAssociatedObject(_ object: Any!,
_ key: UnsafeRawPointer!,
_ value: Any!,
_ policy: objc_AssociationPolicy)
需要注意的是在Swift3.0之前對(duì)于一些C語(yǔ)言的API需要傳void *的指針時(shí)柏肪,swif中對(duì)應(yīng)的是UnsafePointer<Void>叶洞。3.0之后的版本中有了一個(gè)新的類型來(lái)處理這些指針:UnsafeRawPointer
另外貼一個(gè)書(shū)中具體實(shí)現(xiàn)的例子
// MyClass.swift
class MyClass {
}
// MyClassExtension.swift
private var key: Void?
extension MyClass {
var title: String? {
get {
return objc_getAssociatedObject(self, &key) as? String
}
set {
objc_setAssociatedObject(self,
&key, newValue,
.OBJC_ASSOCIATION_RETAIN_NONATOMIC)
}
}
}
// 測(cè)試
func printTitle(_ input: MyClass) {
if let title = input.title {
print("Title: \(title)")
} else {
print("沒(méi)有設(shè)置")
}
}
let a = MyClass()
printTitle(a)
a.title = "tips"
printTitle(a)
這里需要注意的是private var key: Void?
這種寫(xiě)法敛滋。根據(jù)上面runtime里的函數(shù)我們能看出來(lái)key這個(gè)參數(shù)的類型應(yīng)該是const void *
就是一個(gè)指向任意類型常量的指針。在OC中這個(gè)key有多種寫(xiě)法
1替废、static void *key = &key;
2箍铭、static NSString *key = @"key";
3、static char key;
第一種寫(xiě)法可能不太好理解(C語(yǔ)言夠嗆的我掩面走過(guò))椎镣,我查了下了解到這是一個(gè)靜態(tài)無(wú)類型指針key然后用指針的地址賦給key就是一個(gè)初始化了诈火。也就是一個(gè)指向指針的指針。
然后我們可以根據(jù)上面OC中的第二種寫(xiě)法寫(xiě)出對(duì)應(yīng)的Swift版本private var key = "key"
状答,個(gè)人感覺(jué)這種更加直觀可讀冷守。
另外這篇文章中的處理方法是不正確的刀崖。。
Swift中使用C語(yǔ)言指針
具體使用方法可以參考這篇文章以及書(shū)中UnsafePointer這章的相關(guān)內(nèi)容拍摇,這里記錄下一些需要注意的地方亮钦。
- Swift中大體上分
UnsafePointer<Type>
,UnsafeMutablePointer<Type>
兩種數(shù)據(jù)類型對(duì)應(yīng)C中的指針。這個(gè)Type
可以是CInt
,CBool
,CChar
這些類型充活,分別對(duì)應(yīng)C中的相關(guān)類型蜂莉。
UnsafePointer<Type>
對(duì)應(yīng)C中的const Type *
,不允許修改指針的值
UnsafeMutablePointer<Type>
對(duì)應(yīng)Type *
,允許修改 - Swift3.0之前存在
UnsafePointer<Void>
這個(gè)類型混卵,在3.0之后使用UnsafeRawPointer這個(gè)類型來(lái)處理映穗。 - 對(duì)于
&
這個(gè)操作來(lái)說(shuō),它需要一個(gè)var
變量幕随,比如上面例子中的private var key: Void?
,把var
換成let
就會(huì)報(bào)錯(cuò)蚁滋。
cannot pass immutable value as inout argument: 'key' is a 'let' constant
- 當(dāng)需要在Swift3中打印地址的時(shí)候可以參考這篇文章的下面這些方法
// 打印引用類型
func addressHeap<T: AnyObject>(o: T) -> Int {
return unsafeBitCast(o, to: Int.self)
}
// 打印值類型
func address(o: UnsafeRawPointer) -> Int {
return Int(bitPattern: o)
}
unsafeAddressOf()
這個(gè)方法被移除了。