程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> 關於C語言 >> Objective-C 和 Core Foundation 對象相互轉換

Objective-C 和 Core Foundation 對象相互轉換

編輯:關於C語言

Objective-C 和 Core Foundation 對象相互轉換


iOS允許Objective-C 和 Core Foundation 對象之間可以輕松的轉換:

  1. CFStringRef aCFString = (CFStringRef)aNSString;
  2. NSString *aNSString = (NSString *)aCFString;

    針對內存管理問題,ARC 可以幫忙管理 Objective-C 對象, 但是不支持 Core Foundation 對象的管理,所以轉換後要注意一個問題:誰來釋放使用後的對象。 本文重點總結一下類型轉換後的內存管理。



    一、非ARC的內存管理


    倘若不使用ARC,手動管理內存,思路比較清晰,使用完,release轉換後的對象即可。

    1. //NSString 轉 CFStringRef
    2. CFStringRef aCFString = (CFStringRef) [[NSString alloc] initWithFormat:@"%@", string];
    3. //...
    4. CFRelease(aCFString);
    5. //CFStringRef 轉 NSString
    6. CFStringRef aCFString = CFStringCreateWithCString(kCFAllocatorDefault,
    7. bytes,
    8. NSUTF8StringEncoding);
    9. NSString *aNSString = (NSString *)aCFString;
    10. //...
    11. [aNSString release];


      二、ARC下的內存管理


      ARC的誕生大大簡化了我們針對內存管理的開發工作,但是只支持管理 Objective-C 對象, 不支持 Core Foundation 對象。Core Foundation 對象必須使用CFRetain和CFRelease來進行內存管理。那麼當使用Objective-C 和 Core Foundation 對象相互轉換的時候,必須讓編譯器知道,到底由誰來負責釋放對象,是否交給ARC處理。只有正確的處理,才能避免內存洩漏和double free導致程序崩潰。
      根據不同需求,有3種轉換方式 __bridge (不改變對象所有權)

      __bridge_retained 或者 CFBridgingRetain() (解除 ARC 所有權)

      __bridge_transfer 或者 CFBridgingRelease() (

      給予 ARC 所有權)

      1. __bridge_retained 或者 CFBridgingRetain()


      __bridge_retained 或者 CFBridgingRetain() 將Objective-C對象轉換為Core Foundation對象,把對象所有權橋接給Core Foundation對象,同時剝奪ARC的管理權,後續需要開發者使用CFRelease或者相關方法手動來釋放對象。

      來看個例子:
      1. - (void)viewDidLoad
      2. {
      3. [super viewDidLoad];
      4. NSString *aNSString = [[NSString alloc]initWithFormat:@"test"];
      5. CFStringRef aCFString = (__bridge_retained CFStringRef) aNSString;
      6. //正確的做法應該執行CFRelease
      7. //CFRelease(aCFString);
      8. }
        程序沒有執行CFRelease,造成內存洩漏:
        \



        CFBridgingRetain() 是 __bridge_retained 的宏方法,下面兩行代碼等價:

        1. CFStringRef aCFString = (__bridge_retained CFStringRef) aNSString;
        2. CFStringRef aCFString = (CFStringRef) CFBridgingRetain(aNSString);




          2. __bridge_transfer 或者 CFBridgingRelease()


          __bridge_transfer 或者 CFBridgingRelease() 將非Objective-C對象轉換為Objective-C對象,同時將對象的管理權交給ARC,開發者無需手動管理內存。

          接著上面那個內存洩漏的例子,再轉成OC對象交給ARC來管理內存,無需手動管理,也不會出現內存洩漏:
          1. - (void)viewDidLoad
          2. {
          3. [super viewDidLoad];
          4. NSString *aNSString = [[NSString alloc]initWithFormat:@"test"];
          5. CFStringRef aCFString = (__bridge_retained CFStringRef) aNSString;
          6. aNSString = (__bridge_transfer NSString *)aCFString;
          7. }
            CFBridgingRelease() 是__bridge_transfer的宏方法,下面兩行代碼等價:

            1. aNSString = (__bridge_transfer NSString *)aCFString;
            2. aNSString = (NSString *)CFBridgingRelease(aCFString);

              3. __bridge


              __bridge 只做類型轉換,不改變對象所有權,是我們最常用的轉換符。
              從OC轉CF,ARC管理內存:
              1. - (void)viewDidLoad
              2. {
              3. [super viewDidLoad];
              4. NSString *aNSString = [[NSString alloc]initWithFormat:@"test"];
              5. CFStringRef aCFString = (__bridge CFStringRef)aNSString;
              6. } 從CF轉OC,需要開發者手動釋放,不歸ARC管:
                1. - (void)viewDidLoad
                2. {
                3. [super viewDidLoad];
                4. CFStringRef aCFString = CFStringCreateWithCString(NULL, "test", kCFStringEncodingASCII);
                5. NSString *aNSString = (__bridge NSString *)aCFString;
                6. CFRelease(aCFString);
                7. }

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved