引言
Core Foundation框架 (CoreFoundation.framework) 是一組C語(yǔ)言接口竭业,它們?yōu)閕OS應(yīng)用程序提供基本數(shù)據(jù)管理和服務(wù)功能。下面列舉該框架支持進(jìn)行管理的數(shù)據(jù)以及可提供的服務(wù):
1.群體數(shù)據(jù)類(lèi)型 (數(shù)組及舍、集合等)
2.程序包
3.字符串管理
4.日期和時(shí)間管理
5.原始數(shù)據(jù)塊管理
6.偏好管理
7.URL及數(shù)據(jù)流操作
8.線(xiàn)程和RunLoop
9.端口和soket通訊
Core Foundation框架和Foundation框架緊密相關(guān)未辆,它們?yōu)橄嗤δ芴峁┙涌冢獸oundation框架提供Objective-C接口锯玛。如果您將Foundation對(duì)象和Core Foundation類(lèi)型摻雜使用咐柜,則可利用兩個(gè)框架之間的 “toll-free bridging”。所謂的Toll-free bridging是說(shuō)您可以在某個(gè)框架的方法或函數(shù)同時(shí)使用Core Foundatio和Foundation 框架中的某些類(lèi)型攘残。很多數(shù)據(jù)類(lèi)型支持這一特性拙友,其中包括群體和字符串?dāng)?shù)據(jù)類(lèi)型。每個(gè)框架的類(lèi)和類(lèi)型描述都會(huì)對(duì)某個(gè)對(duì)象是否為 toll-free bridged歼郭,應(yīng)和什么對(duì)象橋接進(jìn)行說(shuō)明献宫。
Toll-free bridging 是ARC下OC對(duì)象和Core Foundation對(duì)象之間的橋梁
在開(kāi)發(fā)iOS應(yīng)用程序時(shí)我們有時(shí)會(huì)用到Core Foundation對(duì)象,下面簡(jiǎn)稱(chēng)CF实撒。例如Core Graphics姊途、Core Text涉瘾,并且我們可能需要將CF對(duì)象和OC對(duì)象進(jìn)行相互轉(zhuǎn)化,我們知道捷兰,ARC環(huán)境下立叛,編譯器不會(huì)自動(dòng)管理CF對(duì)象的內(nèi)存,我們需要手動(dòng)管理贡茅。這就是我們?cè)趧?chuàng)建一個(gè)CF對(duì)象以后需要我們使用CFRelease將其手動(dòng)釋放秘蛇。
那么CF和OC相互轉(zhuǎn)化的時(shí)候該如何管理內(nèi)存呢?
我們可以通過(guò)__bridge,__bridge_transfer,__bridge_retained 來(lái)進(jìn)行內(nèi)存管理
1.__bridge
CF和OC對(duì)象轉(zhuǎn)化時(shí)只涉及對(duì)象類(lèi)型不涉及對(duì)象所有權(quán)的轉(zhuǎn)化
//Image I/O 從 NSBundle 讀取圖片數(shù)據(jù)
NSString *imagePath = [[NSBundle mainBundle] pathForResource:@"image" ofType:@"png"];
CGImageSourceRef source = CGImageSourceCreateWithURL((__bridge CFURLRef)[NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"image" ofType:@"png"]], NULL);
如果上面不添加__bridge 顶考,在ARC環(huán)境下赁还,系統(tǒng)會(huì)給出錯(cuò)誤提示和錯(cuò)誤修正,點(diǎn)擊錯(cuò)誤提示的話(huà)驹沿,系統(tǒng)會(huì)為我們自動(dòng)添加__bridge 艘策,因?yàn)樵贠C與CF的轉(zhuǎn)化時(shí)只涉及到對(duì)象類(lèi)型沒(méi)有涉及到對(duì)象所有權(quán)的轉(zhuǎn)化,所以上述代碼不需要對(duì)CF的對(duì)象進(jìn)行釋放渊季,即不需要添加CFRelease
注釋?zhuān)?iOS ARC 和 非ARC 之間的轉(zhuǎn)換方法
1朋蔫,選擇項(xiàng)目中的Targets,選中你所要操作的Target却汉,
2驯妄,選Build Phases,在其中Complie Sources中選擇需要ARC的文件雙擊合砂,
并在輸入框中輸入:-fobjc-arc青扔,如果不要ARC則輸入:-fno-objc-arc
為了解決這一問(wèn)題,我們使用 __bridge 關(guān)鍵字來(lái)實(shí)現(xiàn)id類(lèi)型與void*類(lèi)型的相互轉(zhuǎn)換翩伪。
id obj = [[NSObject alloc] init];
void *p = (__bridge void *)(obj);
NSLog(@"obj retainCount %ld",[(id)p retainCount]);
輸出結(jié)果:
CFDemo[2932:777997] obj retainCount 1
2.__bridge_transfer
常用在CF對(duì)象轉(zhuǎn)化成OC對(duì)象時(shí)赎懦,將CF對(duì)象的所有權(quán)交給OC對(duì)象,此時(shí)ARC就能自動(dòng)管理該內(nèi)存,作用同CFBridgingRelease()
如果非ARC的時(shí)候幻工,我們可能需要寫(xiě)下面的代碼。
// p 變量原先持有對(duì)象的所有權(quán)
id obj = (id)p;
[obj retain];
[(id)p release];
那么ARC有效后黎茎,我們可以用下面的代碼來(lái)替換:
// p 變量原先持有對(duì)象的所有權(quán)
id obj = (__bridge_transfer id)p;
可以看出來(lái)囊颅,__bridge_retained 是編譯器替我們做了 retain 操作,而 __bridge_transfer 是替我們做了 release傅瞻。
3.__bridge_retained
與__bridge_transfer 相反踢代,常用在將OC對(duì)象轉(zhuǎn)化成CF對(duì)象,且OC對(duì)象的所有權(quán)也交給CF對(duì)象來(lái)管理嗅骄,即OC對(duì)象轉(zhuǎn)化成CF對(duì)象時(shí)胳挎,涉及到對(duì)象類(lèi)型和對(duì)象所有權(quán)的轉(zhuǎn)化,作用同CFBridgingRetain()
先來(lái)看使用 __bridge_retained 關(guān)鍵字的例子程序:
id obj = [[NSObject alloc] init];
void *p = (__bridge_retained void *)obj;
此時(shí)retainCount 會(huì)被加1溺森;
從名字上我們應(yīng)該能理解其意義:類(lèi)型被轉(zhuǎn)換時(shí)慕爬,其對(duì)象的所有權(quán)也將被變換后變量所持有窑眯。如果不是ARC代碼,類(lèi)似下面的實(shí)現(xiàn):
id obj = [[NSObject alloc] init];
void *p = obj;
[(id)p retain];
ARC如何獲取retainCount
NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)myObject));
來(lái)來(lái)來(lái)舉個(gè)例子:
NSString *string = [NSString stringWithFormat:@""];
CFStringRef cfString = (__bridge CFStringRef)string;
CFStringRef cfStr = (__bridge_retained CFStringRef)string;
CFRelease(cfString);// 由于Core Foundation的對(duì)象不屬于ARC的管理范疇医窿,所以需要自己release
CFRelease(cfStr);
使用 __bridge_retained 可以通過(guò)轉(zhuǎn)換目標(biāo)處(cfStr)的 retain 處理磅甩,來(lái)使所有權(quán)轉(zhuǎn)移。即使 string 變量被釋放姥卢,cfString變量也變釋放卷要,cfStr 還是可以使用具體的對(duì)象。只是有一點(diǎn)独榴,由于Core Foundation的對(duì)象不屬于ARC的管理范疇僧叉,所以需要自己release。
CFStringRef cfString= CFURLCreateStringByAddingPercentEscapes( NULL, (__bridge CFStringRef)text, NULL, CFSTR("!*’();:@&=+$,/?%#[]"), CFStringConvertNSStringEncodingToEncoding(NSUTF8StringEncoding));
NSString *ocString = (__bridge_transfer CFStringRef)cfString;
所有權(quán)被轉(zhuǎn)移的同時(shí)棺榔,被轉(zhuǎn)換變量將失去對(duì)象的所有權(quán)瓶堕。當(dāng)Core Foundation對(duì)象類(lèi)型向Objective-C對(duì)象類(lèi)型轉(zhuǎn)換的時(shí)候,會(huì)經(jīng)常用到 __bridge_transfer 關(guān)鍵字掷豺。
總結(jié):
- Core Foundation 對(duì)象類(lèi)型不在 ARC 管理范疇內(nèi)
- Cocoa Framework::Foundation 對(duì)象類(lèi)型(即一般使用到的Objectie-C對(duì)象類(lèi)型)在 ARC 的管理范疇內(nèi)
3.__bridge捞烟,__bridge_transfer和__bridge_retained 是CF和OC的橋梁 - 如果不在 ARC 管理范疇內(nèi)的對(duì)象,那么要清楚 release 的責(zé)任應(yīng)該是誰(shuí)以及各種對(duì)象的生命周期是怎么樣的
這是目前在學(xué)習(xí)ImageIO蘋(píng)果官方提供的圖片解碼器当船,遇到OC和CF之間相互轉(zhuǎn)化的一些問(wèn)題题画,重新整理,細(xì)致的了解下德频,以方便后期更加深入的學(xué)習(xí)苍息。
PS:來(lái)簡(jiǎn)書(shū)混,關(guān)注是必須的壹置,點(diǎn)贊?? 是要給的竞思!