簡介
Foundation與Core Foundation是兩個關(guān)聯(lián)密切的框架,區(qū)別主要在于Foundation是以O(shè)bjective-C實現(xiàn)的,Core Foundation是以C語言實現(xiàn)的。在編程中有時會遇到兩個框架內(nèi)做類型轉(zhuǎn)換,比如NSDictionary與CFDictionary之間的轉(zhuǎn)換。
MRC
在MRC模式下,由于id類型其實也是指向結(jié)構(gòu)體的一個指針,所以可以像C語言一樣做類型轉(zhuǎn)換,自己做好內(nèi)存管理即可。如下:
NSDictionary *nDict = @{@"key" : @"value"};
CFDictionaryRef cDict = nDict;
NSDictionary *nDictOther = cDict;
ARC
在ARC模式下,編譯器會為我們做內(nèi)存管理,但是在遇到這種類型轉(zhuǎn)換的時候,編譯器無法判斷該怎么處理這種對象的內(nèi)存管理,即ARC沒有辦法處理Core Foundation框架下的內(nèi)存管理問題。所以便有了下面幾個關(guān)鍵字:
__bridge
__bridge_retained
__bridge_transfer
上述三個關(guān)鍵字的區(qū)別在于對象的內(nèi)存管理權(quán)的處理
| 名稱 | 描述 |
|---|---|
| __bridge | 只做類型轉(zhuǎn)換,內(nèi)存管理權(quán)不變 |
| __bridge_retained | 將Foundation對象轉(zhuǎn)換為CoreFoundation對象,交出內(nèi)存管理權(quán),需要做CFRelease操作,避免內(nèi)存泄漏 |
| __bridge_transfer | 將CoreFoundation對象轉(zhuǎn)為Foundation對象,內(nèi)存管理權(quán)交出,歸ARC處理 |
__bridge
NSDictionary *dict = @{@"key" : @"value"};
CFDictionaryRef cfDict = (__bridge CFDictionaryRef)dict;
dict = nil;
NSLog(@"%@", cfDict);//這時這里會出現(xiàn)EXC_BAD_ACCESS,即訪問了野指針。
但是使用NSString時需要注意:
NSString *nsString = @"123";
CFStringRef cfString = (__bridge NSString *)nsString;
nsString = nil;
NSLog(@"%@", cfString);//使用string的這個例子,這么用沒有問題
相當(dāng)于__unsafe_unretained,nsString被設(shè)置為nil后,cfString會變成野指針?需確定。
經(jīng)過確定,只使用__bridge時,的確會在原oc變量釋放后cf對象變成野指針的情況,但是不能用NSString這個例子來講,因為NSString的引用計數(shù)受到系統(tǒng)的特殊處理,具體參考鏈接。
__bridge_retained
NSDictionary *nDdict = @{@"key" : @"value"};
CFDictionaryRef cDict = (__bridge_retained CFDictionaryRef)nDict;
nDict = nil;
NSLog(@"%@", cDict);//這時是沒問題的
...
CFRelease(cDict);
nDict不再持有內(nèi)存的管理權(quán),nDict即使設(shè)置為nil,也不會影響cDict對內(nèi)存的引用,需要開發(fā)者找個合適的時機去調(diào)用CFRelease(cDict)來釋放。
__bridge_transfer
CFDictionaryRef cDict = CFDictionaryCreate(CFAllocatorGetDefault(), &("key"), &("value"), 1, NULL, NULL);
NSDictionary *nDict = (__bridge_transfer NSDictionary *)cDict;
//nDict = CFBridgingRelease(cDict);也可以直接用這個函數(shù)
cDict對內(nèi)存的管理權(quán)移交給nDict,即ARC來處理。