MRC 环境下

  id 变量赋值给 void * 变量运行时不会有问题。

id obj1 = [NSObject new];void * p = obj1;

  void * 变量赋值给 id 变量并调用其实例方法,运行时也不会有问题。

id obj2 = p;
[obj2 release];

ARC 环境下

  直接赋值报错

  

  系统给出解决方案:

  

__bridge

{  id obj1 = [NSObject new];
  void * p = (__bridge void *)obj1;
  id obj2 = (__bridge id)p;} 

  id 变量赋值给 void * 变量时的__bridge 与 __unsafe_unretained 修饰符相近,甚至会更低。如果管理时不注意 id 对象的持有者,就会因悬垂指针而导致程序崩溃。

  PS:指针指向曾经存在的对象,但该对象现在不存在了,那么该指针即为悬垂指针

  __bridge 不持有对象。

  在代码中加入了 dict = nil 运行时会 crash。如下:

{  NSDictionary * dict = @{ @"k": @"v" };
  void * p = (__bridge void *)(dict);
  dict = nil;
  NSLog(@"%@", p);            }

  __bridge 还有另外两种转换:__bridge_retained、__bridge_transfer。

__bridge_retained 

  __bridge_retained 转换会导致被赋值的变量也持有所赋值的对象,等同于 MRC 环境下使用的 retain 方法。MRC 环境下使用无效果。

  MRC 环境下写法: 

{  NSDictionary * dict = @{ @"k": @"v" };
  void * p = [dict retain];    // dict.retainCount = 2}

  ARC 环境下写法:

{
  NSDictionary * dict = @{ @"k": @"v" };
  void * p = (__bridge_retained void *)(dict);   // dict.retainCount = 2
}

__bridge_transfer

  __bridge_transfer 转换与 __bridge_retained 行为相反,原有的变量在通过 __bridge_transfer 赋值给目标变量后引用计数减一,等同于 MRC 环境下使用的 release 方法。MRC 环境下使用无效果。

  MRC 环境下写法:

{  const void * keys[] = {};
  const void * values[] = {};

  CFDictionaryRef cf = CFDictionaryCreate(kCFAllocatorDefault, keys, values, , NULL, NULL);  // retainCount = 1
  NSDictionary * p = (__bridge NSDictionary *)(cf);  // retainCount = 2
  CFRelease(cfDict);                     // retainCount = 1
  NSLog(@"%d", CFGetRetainCount(cfDict));}

  ARC 环境下写法:

{
  const void * keys[] = {};
  const void * values[] = {};

  CFDictionaryRef cf = CFDictionaryCreate(kCFAllocatorDefault, keys, values, 1, NULL, NULL);  NSLog(@"%d", CFGetRetainCount(cf));                  // retainCount = 1  NSDictionary * dict = (__bridge_transfer NSDictionary *)cf;   // retainCount + 1 - 1  NSLog(@"%d", CFGetRetainCount(cf));                  // retainCount = 1  NSLog(@"%@", cf);                         
}

  原本 CFBridgingRelease() 会导致 cf 对象的引用计数 - 1,但因为 dict 指针是强引用,所以最终成了先引用计数 + 1,然后引用计数 - 1,对象的引用计数还是 1。

  注意:引用计数是对象的属性,不是指针。

Objective-C 对象与 CoreFoundation 对象

  这些转换多数用于 Objective-C 对象与 Core Foundation 对象之间。

  Core Foundation 对象主要使用在用 C 语言编写的 CoreFoundation.framework 中,并使用引用计数的对象。两者对引用计数的操作方法:

Objective-C Core Foundation Effect
retain CFRetain() retainCount + 1
release CFRelease() retainCount - 1
retainCount CFGetRetainCount()  

  Core Foundation 对象与 Objective-C 对象不同之处只在于是由 CoreFoundation.framework 还是 Foundation.framework 所生成的。无论是由哪种框架生成的对象,都能在不同的框架中使用。Foundation.framework 的 api 生成并持有的对象可以用 CoreFoundation.framework 的 api 释放。当然,反过来也是可以的。

  MRC 环境下只用简单的 C 语言的转换也能实现互换。另外这种转换不需要使用额外的 CPU 资源,因此也被称为"免费桥"(Toll-FreeBridge)。如下函数:

    CFTypeRef CFBridgingRetain(id X)  {   return (__bridge_retained CFTypeRef)X;  }   

    id CFBridgingRelease(CFTypeRef X) {   return (__bridge_transfer id)X;   }

{  NSDictionary * dict = (@{ @"k": @"v" });
  CFDictionaryRef cf = CFBridgingRetain(dict);
  CFShow(cf);
  NSLog(@"%d", dict.retainCount);  // 2     dict.retainCount = CFGetRetainCount(cf)  CFRelease(cf);  NSLog(@"%d", dict.retainCount);  // 1}

  由此可知,Objective-C 对象能够作为 Core Foundation 对象来使用。也可以通过 CFRelease 来使引用计数减一。当然,也可以使用 __bridge_retained 转换来替代 CFBridgingRetain()。大家可选用自己更熟悉的方法。

CFDictionaryRef cf = (__bridge_retained CFDictionaryRef)dict;

  这次反过来,将使用 Core Foundation 的 api 生成并持有对象,将该对象作为 Objective-C 对象来处理。

{
  CFMutableArrayRef cfObject = CFArrayCreateMutable(kCFAllocatorDefault, , NULL);
  NSLog(
  NSArray * arr = CFBridgingRelease(cfObject);           // retainCount + 1 - 1
  NSLog(@"retain count = %d", CFGetRetainCount(cfObject));    // 1
  NSLog(@"%@", arr);
}

参考文章:http://book.2cto.com/201305/23864.html

id 与 void * 转换的更多相关文章

  1. ARC 类型转换:显式转换 id 和 void *

    http://blog.csdn.net/chinahaerbin/article/details/9471419 /* * ARC有效时三种类型转换: */ 1.__bridge           ...

  2. VC中句柄、指针、ID之间的转换

    win32直接操作的是句柄HANDLE,每个句柄就对应windows窗口,而vc对HANDLE进行类封装,间接操作的都是HANDLE,现在句柄只是类的一个成员变量. 从句柄到指针 CWnd* pWnd ...

  3. 【转载】混编ObjectiveC++

    原文:混编ObjectiveC++ 最近有点炒冷饭的嫌疑,不过确实以前没有Git Or Blog的习惯,所以很多工作上的技术分享就存留在了电脑的文档里,现在还是想重新整理一下,再分享出来. 混编C++ ...

  4. 新浪微博id的62进制转换

    某条微博链接 某条微博的链接如下,同样省略了后面的无关参数 http://weibo.com/2803301701/CeaOU15IT CeaOU15IT为这条微博的mid,与之相对应的还有一个id, ...

  5. org.Hs.eg.db包简介(转换NCBI、ensemble等数据库中基因ID,symbol等之间的转换)

    1)安装载入 ------------------------------------------- if("org.Hs.eg.db" %in% rownames(install ...

  6. json转换成对象

    在json转换成对象时,json的key会与java 类的字段一一对应.如果没有映射上的java字段会在该数据类型上填充默认值,如int 0,String null 等. 没有映射的json key在 ...

  7. Json转换利器Gson之实例一-简单对象转化和带泛型的List转化 (转)

    Gson 是 Google 提供的用来在 Java 对象和 JSON 数据之间进行映射的 Java 类库.可以将一个 JSON 字符串转成一个 Java 对象,或者反过来. jar和源码下载地址: h ...

  8. BlocksKit初见:一个支持将delegate转换成block的Cocoa库

    简介 项目主页: https://github.com/zwaldowski/BlocksKit BlocksKit 是一个开源的框架,对 Cocoa 进行了扩展,将许多需要通过 delegate 调 ...

  9. SpringMvc 相关,bean map转换,百度天气,base64.js,jsBase64.java;

    1. Map<String, Object>与JavaBean[POJO, Model]转换; //model public class model{ private int id; pr ...

随机推荐

  1. 移动cup

    intel处理器M和U H结尾的有什么具体区别 笔记本CPU 酷睿i 系列U M H详解: U 低压版(低电压-性能消减)最低主频:1.7-1.9GHZ M 准电压(笔记本上的电压标准)最低主频:2. ...

  2. 迷你MVVM框架 avalonjs 1.3.6发布

    本版本是一次重要的升级,考虑要介绍许多东西,也有许多东西对大家有用,也发到首页上来了. 本来是没有1.36的,先把基于静态收集依赖的1.4设计出来后,发现改动太多,为了平缓升级起见,才减少了一部分新特 ...

  3. Ansible 从MySQL数据库添加或删除用户

    mysql_user - 从MySQL数据库添加或删除用户. 概要 要求(在执行模块的主机上) 选项 例子 笔记 状态 支持 概要 从MySQL数据库添加或删除用户. 要求(在执行模块的主机上) My ...

  4. springboot重定向

    参考https://www.cnblogs.com/kxkl123/p/7800967.html public String test() { return "redirect:/" ...

  5. TZOJ 4746 Xiangqi(模拟棋盘数组)

    描述 Xiangqi is one of the most popular two-player board games in China. The game represents a battle ...

  6. 源码安装php时出现Sorry, I cannot run apxs. Possible reasons follow:

    1.可能的原因是你没有安装perl > yum install perl > yum install httpd-devel 2.在你apache安装目录下的bin下找到apxs,并用vi ...

  7. CMakeList 编写规则 -1

    CMAKE 常见指令 CMAKE_MINIMUM_REQUIRED(VERSION 2.8) PROJECT(XXX) SET(CMAKE_BUILD_TYPE Release) SET(CMAKE_ ...

  8. web项目传classes目录项目正常,打包成jar不能运行。

    笔者最近使用tomcat9,由于工作洁癖,盯上了tomcat启动日志里的"No TLD files were found in"字样,如下 15-Sep-2017 02:19:09 ...

  9. Java ENUM枚举的用法

    DK1.5引入了新的类型——枚举.在 Java 中它虽然算个“小”功能,却给我的开发带来了“大”方便. 用法一:常量 在JDK1.5 之前,我们定义常量都是: publicstaticfianl... ...

  10. BZOJ1221 [HNOI2001]软件开发 - 费用流

    题解 非常显然的费用流. 但是建图还是需要思考的QuQ 将每天分成两个节点 $x_{i,1}, x_{i,2} $, $ x_{i,1}$用于提供服务, $x_{i ,2}$ 用来从源点获得$nd[i ...