我正在尝试在同时使用CoreFoundation和Foundation的Linux上编译一些代码,但是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和CF类型之间进行转换的协议(protocol),许多类型具有初始化程序和属性,但是它们都是internalprivate。我可以只使用CFStringCreateWithCString,但这需要与其他类对一起使用,例如InputStreamCFReadStream

我在这里缺少什么吗,还是真的没有办法在Linux上的Foundation和CoreFoundation类型之间进行转换?

最佳答案

好像我可以unsafeBitCastNSString或从CFString进行ojit_code:

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类型被设计为具有相同的内存布局-这就是为什么免费电话桥接有效的原因。我很惊讶它可以与Foundation的Swift实现一起使用。

关于linux - 在Linux上往返于CoreFoundation,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/39838575/

10-09 16:28
查看更多