我试图在Linux上编译一些使用CoreFoundation和Foundation的代码,但是Linux并没有像MacOS和iOS那样实现桥接。
Objective-C和Swift之间的桥接工作:
import Foundation import CoreFoundation import Glibc func wantsNSString(_ string: NSString) { print(string) } let string = "Hello, world!" wantsNSString(string._bridgeToObjectiveC())
但我不知道如何桥接到CoreFoundation。 我不能只传递一个NSString
到一个需要CFString
的函数:
import Foundation import CoreFoundation import Glibc func wantsCFString(_ string: CFString) { print(string) } let string = "Hello, world!" wantsCFString(string._bridgeToObjectiveC()) //error: cannot convert value of type 'String._ObjectType' (aka 'NSString') to expected argument type 'CFString'
我不能像在macOS上那样施放它:
import Foundation import CoreFoundation import Glibc func wantsCFString(_ string: CFString) { print(string) } let string = "Hello, world!" wantsCFString(string._bridgeToObjectiveC() as CFString) //error: 'String._ObjectType' (aka 'NSString') is not convertible to 'CFString'; did you mean to use 'as!' to force downcast?
as!
像错误消息提示编译,但导致运行时崩溃( Illegal instruction
),并as?
产生错误:
error: conditional downcast to CoreFoundation type 'CFString' will always succeed
Bridging.swift
有在NS和CFtypes之间转换的协议,许多types都有初始值和属性,但都是internal
或private
。 我只能使用CFStringCreateWithCString
,但是这需要与其他类对(如InputStream
和CFReadStream
。
我在这里丢失了什么东西,还是真的没有办法在Linux上的Foundation和CoreFoundationtypes之间进行转换?
它看起来像我可以unsafeBitCast
NSString
/从CFString
:
import Foundation import CoreFoundation import Glibc func wantsCFString(_ string: CFString) { print(string) } let string = "Hello, world!" wantsCFString(unsafeBitCast(string._bridgeToObjectiveC(), to: CFString.self) //prints "Hello, world!"
这是有道理的,因为CoreFoundation和Foundation类型被设计为具有相同的内存布局 – 这就是免费桥接工程的原因。 虽然我对基金会的Swift实现起作用,但有些惊讶。