ARC中__bridge, __bridge__transfer, __bridge_retained 关系

33 篇文章 0 订阅
32 篇文章 0 订阅

现在已经凌晨了,但是我却没发觉困。

琢磨了好久的东西,终于顿悟。

这篇是其中的一个点,记录下来,以备后续查阅!


说到__bridge,就不得不说Objective-C和Core Foundation对象之间的关系。

当你在 Objective-C  Core Foundation 对象之间进行转换时,就需要使用 Bridge cast。

今天的多数应用很少需要使用 Core Foundation,大多数工作都可以直接使  Objective-C 类来完成。但是某些底层 API,如 Core Graphics  Core Text, 都基于 Core Foundation,而且不太可能会有 Objective-C 的版本。幸运的是, iOS 的设计使得这两种类型的对象非常容易转换。

例如 NSString  CFStringRef 就可以同等对待,在任何地方都可以互换使 用,背后的设计就是toll-free bridging。在 ARC 之前,只需要使用一个简单 的强制类型转换即可:

当然,alloc 分配了 NSString 对象,你需要在使用完之后进行释放,注意是 释放转换后的CFStringRef 对象:
CFRelease(s1);

反过来,从 Core Foundation  Objective-C 的方向也类似:

现在我们使用了 ARC,情况变得不一样!以下代码在手动内存管理中是完全 合法的,但在 ARC 中却存在问题:

- (NSString *)escape:(NSString *)text
{

CFStringRef s1 = (CFStringRef) [[NSString alloc]initWithFormat:@"Hello, %@!", name];

CFStringRef s2 = CFStringCreateWithCString(kCFAllocatorDefault, bytes,kCFStringEncodingMacRoman);

NSString *s3 = (NSString *)s2;

	[s3 release];

return [(NSString *)CFURLCreateStringByAddingPercentEscapes(NULL,(CFStringRef)text,NULL,(CFStringRef)@"!*'();:@&=+$,/?%#[]", 

    CFStringConvertNSStringEncodingToEncoding(NSUTF8StringEncoding))autorelease];// 这里不需要 bridgingcasts,因为这是    一个常量,不需要释放!

}

首先需要移除 autorelease 调用。然后编译器还会报两个类型转换错误:

Cast of C pointer type 'CFStringRef' to Objective-C pointer type 'NSString *' requires a bridged cast

Cast of Objective-C pointer type 'NSString *' to C pointer type 'CFStringRef' requires a bridged cast

编译器必须知道由谁来负责释放转换后的对象,如果你把一个 NSObject 当 作 Core Foundation对象来使用,则 ARC 将不再负责释放该对象。但你必须明确 地告诉 ARC 你的这个意图,编译器没办法自己做主。同样如果你创建一个 Core Foundation 对象并把它转换为 NSObject 对象,你也必须告诉 ARC 占据对象的所 有权,并在适当的时候释放该对象。这就是所谓的 bridging casts。

CFURLCreateStringByAddingPercentEscapes()函数的参数需要两个 CFStringRef 对象,其中常量NSString 可以直接转换,因为不需要进行对象释 放;但是 text 参数不一样,它是传递进来的一个NSString 对象。而函数参数 和局部变量一样,都是 strong 指针,这种对象在函数入口处会被retain,并且 对象会持续存在直到指针被销毁(这里也就是函数返回时)。

对于 text 参数,我们希望 ARC 保持这个变量的所有权,同时又希望临时将 它当作 CFStringRef 对象来使用。这种情况下可以使用__bridge 说明符,它告 诉 ARC 不要更改对象的所有权,按普通规则释放该对象即可。

多数情况下,Objective-C 对象和 Core Foundation 对象之间互相转换时, 我们都应该使用__bridge。但是有时候我们确实需要给予 ARC 某个对象的所有权, 或者解除 ARC 对某个对象的所有权。这种情况下我们就需要使用另外两种 bridging casts:

  • __bridge_transfer:给予 ARC 所有权

  • __bridge_retained:解除 ARC 所有权

在上面代码中,"return (NSString *)CFURLCreateStringByAddingPercentEscapes",编译器弹出的修复提示有两个:

  • 两个解决办法:__bridge 和 __bridge_transfer,正确的选择应该  __bridge_transfer。

    因为 CFURLCreateStringByAddingPercentEscapes() 函数创建了一个新 的 CFStringRef 对象,当然我们要的是 NSString 对象,因此我们使用了强制转 换。实际上我们真正想要做的是:

     CFURLCreateStringByAddingPercentEscapes 函数的 create 可以看 出,函数会返回一个retain 过的对象。某个人需要负责在适当的时候释放该对 象,如果我们不把这个对象返回为NSString,则通常我们需要自己调用:

    {

    }

    不过 ARC 只能作用于 Objective-C 对象,不能释放 Core Foundation 对象。 因此这里你仍然需要调用 CFRelease()来释放该对象。

    这里我们的真实意图是:转换新创建的 CFStringRef 对象为 NSString 对象, 并且当我们不再需要使用这个 NSString 对象时,ARC 能够适当地释放它。

    因此我们使用 __bridge_transfer告诉ARC:"嘿!ARC,这个CFStringRef 对象现在是一个NSString 对象了,我希望你来销毁它,我这里就不调用 CFRelease()来释放它了"。

    如果我们使用 __bridge,就会导致内存泄漏。ARC并不知道自己应该在使 用完对象之后释放该对象,也没有人调用 CFRelease()。结果这个对象就会永远 保留在内存中。因此选择正确的bridge 说明符是至关重要的。

    为了代码更加可读和容易理解,iOS 还提供了一个辅助函数: CFBridgingRelease()。函数所做事情和 __bridge_transfer 强制转换完全一 样,但更加简洁和清晰。CFBridgingRelease() 函数定义为内联函数,因此不会 导致额外的开销。函数之所以命名为 CFBridgingRelease(),是因为一般你会在 需要使用 CFRelease()释放对象的地方,调用 CFBridgingRelease()来传递对象 的所有权。

    因此最后我们的代码如下:

    - (NSString *)escape:(NSString *)text

    {

    }

    另一个常见的需要 CFBridgingRelease 的情况是 AddressBook framework:

    - (NSString *)firstName

    {

    }

    只要你调用命名为 Create, Copy, Retain  Core Foundation 函数,你都 需要使用CFBridgingRelease()安全地将值传递给ARC。

    __bridge_retained 则正好相反,假设你有一个 NSString 对象,并且要将 它传递给某个 Core Foundation API,该函数希望接收这个 string 对象的所有 权。这时候你就不希望 ARC 也去释放该对象,否则就会对同一对象释放两次,而 且必将导致应用崩溃!换句话说,使用__bridge_retained 将对象的所有权给 予 Core Foundation,而 ARC 不再负责释放该对象。

    如下面例子所示:

    return CFBridgingRelease(CFURLCreateStringByAddingPercentEscapes(NULL,(__bridge CFStringRef)text,NULL,CFSTR("!*'();:@&=+$,/?%#[]"),

     CFStringConvertNSStringEncodingToEncoding(NSUTF8StringEncoding)));

    return CFBridgingRelease(ABRecordCopyCompositeName(...));

    NSString *s1 = [[NSString alloc] initWithFormat:@"Hello, %@!", name];

    CFStringRef s2 = (__bridge_retained CFStringRef)s1;

    // do something with s2
    CFRelease(s2);

     

一旦 (__bridge_retained CFStringRef) 转换完成,ARC 就不再负责释放该 对象。如果你在这里使用 __bridge,应用就很可能会崩溃。ARC 可能在 Core Foundation 正在使用该对象时,释放掉它。

同样__bridge_retained 也有一个辅助函数:CFBridgingRetain()。从名字 就可以看出,这个函数会让 Core Foundation 执行 retain,实际如下:

现在你应该明白了,上面例子的 CFRelease()是和 CFBridgingRetain()对应 的。你应该很少需要使用__bridge_retained 或 CFBridgingRetain()。

__bridge 转换不仅仅局限于 Core Foundation 对象,某些 API 使用 void * 指针作为参数,允许你传递任何东西的引用:Objective-C 对象、Core Foundation 对象、malloc()内存缓冲区等等。void *表示这是一个指针,但实际的数据类型 可以是任何东西!

要将 Objective-C 对象和 void *互相转换,你也需要使用__bridge 转换, 如下:

 animation delegate 方法中,你再将对象强制转回来:

总结:

  • 使用 CFBridgingRelease(),从 Core Foundation 传递所有权给 Objective-C;

  • 使用 CFBridgingRetain(),从 Objective-C 传递所有权给 Core Foundation;

  • 使用__brideg,表示临时使用某种类型,不改变对象的所有权。 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用桥接(bridge)将NSString对象转换为CFStringRef类型,然后使用CFStringGetCString或CFStringGetCharacters等函数将其转换为C语言字符串或Unicode字符数组。例如: ``` NSString *str = @"Hello, world!"; CFStringRef cfStr = (__bridge CFStringRef)str; const char *cStr = CFStringGetCStringPtr(cfStr, kCFStringEncodingUTF8); ``` 注意,CFStringRef和NSString并不是相同的类型,它们只是在Core Foundation和Foundation框架相互转换的对象。因此,在使用CFStringRef时需要遵循Core Foundation的内存管理规则,如需要手动释放内存等。 ### 回答2: 要将NSString*转换为struct __CFString *,可以使用桥接(bridge)方法来实现。 在Objective-C,NSString和CFString是可以相互转换的,因为它们彼此兼容。所以我们可以使用__bridge动态转换不同的指针类型。 具体步骤如下: 1. 首先,引入CoreFoundation框架,因为CFString是属于CoreFoundation的一部分。在文件头部添加以下代码: ```objc #import <CoreFoundation/CoreFoundation.h> ``` 2. 在需要进行转换的地方,使用__bridge关键字进行类型转换。代码如下: ```objc NSString *str = @"Hello, World!"; struct __CFString *cfString = (__bridge struct __CFString *)str; ``` 这样,通过将NSString指针转换为struct __CFString指针,就可以实现类型的转换。 需要注意的是,这种转换是一个桥接操作,不会进行实际数据的复制,只是让两种类型的指针相互引用同一块内存区域。 另外,为了避免出现内存管理问题,如果CFString在ARC环境下使用,还需要使用__bridge_retained进行转换,确保CFString对象在不再使用时被正确释放。 总结起来,NSString*转换为struct __CFString*的关键在于使用__bridge关键字进行转换操作。 ### 回答3: NSString*转换为struct __CFString *的方法是通过桥接(bridging)操作来实现的。在Objective-C,可以使用__bridge关键字将NSString对象转换为指向CFStringRef的指针。 具体转换方法如下: 1. 首先,要确保项目引入了CoreFoundation框架。 2. 使用__bridge关键字将NSString对象转换为struct __CFString *。代码示例如下: NSString *myString = @"Hello World"; struct __CFString *myCFString = (__bridge struct __CFString *)myString; 在上述代码,myString是一个NSString对象,将其转换为struct __CFString *类型的指针myCFString。 3. 进行转换后,可以使用CFStringRef的函数和方法来操作myCFString。例如,可以使用CFStringGetLength()函数来获取myCFString的长度。 CFIndex length = CFStringGetLength(myCFString); 需要注意的是,转换后的指针只是一个指向NSString对象底层CoreFoundation框架的数据结构的指针,并不是一个完全独立的对象。因此,在操作完myCFString后,不需要手动释放内存,系统会自动处理内存管理。 总之,通过使用__bridge关键字,可以方便地将NSString*转换为struct __CFString *类型的指针,从而在需要使用CoreFoundation框架的场景使用NSString对象。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值