一个对象指针(即一个引用类型的实例)可以是
转换为UnsafePointer<Void>(Swift 3 中const void *、UnsafeRawPointer 的Swift 映射)并返回。在 Objective-C 中你会写
void *voidPtr = (__bridge void*)self;
//
MyType *mySelf = (__bridge MyType *)voidPtr;
(请参阅 Clang ARC 文档中的 3.2.4 Bridged casts 了解这些的确切含义
演员表。)
为此,Swift 有一个 Unmanaged 类型。
使用起来有点麻烦,因为它适用于COpaquePointer
而不是UnsafePointer<Void>。这里有两个辅助方法
(以 Objective-C __bridge 演员命名):
func bridge<T : AnyObject>(obj : T) -> UnsafePointer<Void> {
return UnsafePointer(Unmanaged.passUnretained(obj).toOpaque())
// return unsafeAddressOf(obj) // ***
}
func bridge<T : AnyObject>(ptr : UnsafePointer<Void>) -> T {
return Unmanaged<T>.fromOpaque(COpaquePointer(ptr)).takeUnretainedValue()
// return unsafeBitCast(ptr, T.self) // ***
}
“复杂”的表达式只需要满足 Swifts
严格的类型系统。在编译后的代码中,这只是一个演员表
指针之间。 (它可以写得更短,如*** cmets 所示
如果您愿意使用“不安全”的方法,但编译
代码相同。)
使用这个辅助方法,您可以将 self 传递给 C 函数
let voidPtr = bridge(self)
(或 UnsafeMutablePointer<Void>(bridge(self)) 如果 C 函数需要
可变指针),并将其转换回对象指针——例如
在回调函数中 - as
let mySelf : MyType = bridge(voidPtr)
不会发生所有权转移,因此您必须确保self
只要使用 void 指针就存在。
为了完整起见,Objective-C 中 __bridge_retained 和 __bridge_transfer 的 Swift 等效项是
func bridgeRetained<T : AnyObject>(obj : T) -> UnsafePointer<Void> {
return UnsafePointer(Unmanaged.passRetained(obj).toOpaque())
}
func bridgeTransfer<T : AnyObject>(ptr : UnsafePointer<Void>) -> T {
return Unmanaged<T>.fromOpaque(COpaquePointer(ptr)).takeRetainedValue()
}
bridgeRetained() 将对象指针转换为 void 指针并
保留对象。 bridgeTransfer() 转换
void 指针返回一个对象指针并消耗保留。
一个优点是对象不能在
因为持有强引用而调用。缺点是
调用必须适当平衡,并且很容易导致保留
循环。
Swift 3 (Xcode 8) 更新:
func bridge<T : AnyObject>(obj : T) -> UnsafeRawPointer {
return UnsafeRawPointer(Unmanaged.passUnretained(obj).toOpaque())
}
func bridge<T : AnyObject>(ptr : UnsafeRawPointer) -> T {
return Unmanaged<T>.fromOpaque(ptr).takeUnretainedValue()
}
func bridgeRetained<T : AnyObject>(obj : T) -> UnsafeRawPointer {
return UnsafeRawPointer(Unmanaged.passRetained(obj).toOpaque())
}
func bridgeTransfer<T : AnyObject>(ptr : UnsafeRawPointer) -> T {
return Unmanaged<T>.fromOpaque(ptr).takeRetainedValue()
}
“不安全指针”的相关变化在
中有所描述