Toll-Free Bridged Types
在 Core Foundation 框架和 Foundation 框架中有很多数据类型可以交替转换。能够被交替转换的数据类型也被叫做 Toll-Free Bridged 数据类型。这意味着你能像参数一样使用相同的数据结构对一个 Core Foundation 的函数进行调用,或者像 Objective-C 的消息接受模式一样执行。例如, NSLocale (查看NSLocale Class Reference)可以与在 Core Foundation 中对应的 CFLocale (查看CFLocale Reference)之间互相转换。
不是所有数据类型都是 Toll-Free Bridged ,即使它们的名字可能让你认为它们是。例如, NSRunLoop 是没有对应的桥接类型 CFRunLoop , NSBundle 也没有对应的桥接类型 CFBundle , NSDateFormatter 同样没有对应的桥接类型 CFDateFormatter 。
文章末尾表 1 提供了一份支持无缝桥接的数据类型的列表。
注意:假如你使用一个自定义回调在一个
Core Foundation框架的集合中,包含一个NULL回调,当使用Objective-C的方式接入它,它的内存管理方式是未定义的。
类型转换和对象语义周期声明
通过无缝桥接技术,在一个你以 NSLocale * 做为一个参数的方法的例子中,你能传递一个 CFLocaleRef 结构体,并且当你看到有一个 CFLocaleRef 参数的函数中,你能够传递一个 NSLocale 实例对象。当然你也必须提供给编译器相关的一些其它信息:第一,你必须转换一种类型成其它;第二,你可能必须指明对象的语义生命周期。
编译器理解 Objective-C 的方法并且返回 Core Foundation 数据类型,下面是 Cocoa 命名转换的历史(查看Advanced Memory Management Programming Guide)。例如,编译器知道,在 iOS 中,通过 UIColor 的 CGColor 方法返回的 CGColor 并不应该被持有。你必须使用恰当的类型转换,像下面例子演示的那样:
1 | NSMutableArray *colors = [NSMutableArray arrayWithObject:(id)[[UIColor darkGrayColor] CGColor]]; |
编译器并不会自动管理 Core Foundation 对象的生命周期。你必须告诉编译器对象的语义所属关系通过使用一种转换(定义在objc/runtime.h)或者 Core Foundation 风格的宏(定义在 NSObject.h):
__bridge关键字表示转换指针在Objective-C和Core Foundation之间而不会转换所属关系。__bridge_retained关键字或者CFBridgingRetain表示转换指针在Objective-C和Core Foundation之间并且把所属权交给你。你负责调用CFRelease或者相关的函数来交出对象的所属权。__bridge_transfer关键字或者CFBridgingRelease表示转换一个非Objective-C的指针到Objective-C并且转换所属权给ARC。ARC负责交出对象的所属权。
下面是一些例子:
1 | NSLocale *gbNSLocale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_GB"]; |
下面的例子显示了使用口述的 Core Foundation 内存管理规则来管理 Core Foundation 内存:
1 | - (void)drawRect:(CGRect)rect { |
无缝桥接类型
下面的表格提供了一个在 Core Foundation 和 Foundation 中可以交替转换数据类型列表。对每一对桥接类型,表也列举出了这些无缝桥接类型在 OS X 中的可用版本。
| Core Foundation 类型 | Foundation 类型 | 可用性 |
|---|---|---|
| CFArrayRef | NSArray | OS X v10.0 |
| CFAttributedStringRef | NSAttributedString | OS X v10.4 |
| CFCalendarRef | NSCalendar | OS X v10.0 |
| CFCharacterSetRef | NSCharacterSet | OS X v10.4 |
| CFDataRef | NSData | OS X v10.0 |
| CFDateRef | NSDate | OS X v10.4 |
| CFDictionaryRef | NSDictionary | OS X v10.0 |
| CFErrorRef | NSError | OS X v10.4 |
| CFLocaleRef | NSLocale | OS X v10.0 |
| CFMutableArrayRef | NSMutableArray | OS X v10.4 |
| CFMutableAttributedStringRef | NSMutableAttributedString | OS X v10.0 |
| CFMutableCharacterSetRef | NSMutableCharacterSet | OS X v10.4 |
| CFMutableDataRef | NSMutableData | OS X v10.0 |
| CFMutableDictionaryRef | NSMutableDict | OS X v10.4 |
| CFMutableSetRef | NSMutableSet | OS X v10.0 |
| CFMutableStringRef | NSMutableString | OS X v10.4 |
| CFNumberRef | NSNumber | OS X v10.0 |
| CFReadStreamRef | NSInputStream | OS X v10.4 |
| CFRunLoopTimerRef | NSTimer | OS X v10.0 |
| CFSetRef | NSSet | OS X v10.4 |
| CFStringRef | NSString | OS X v10.0 |
| CFTimeZoneRef | NSTimeZone | OS X v10.4 |
| CFURLRef | NSURL | OS X v10.0 |
| CFWriteStreamRef | NSOutputStream | OS X v10.4 |