id 与 void * 转换

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, 1, 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, 0, NULL);
  NSLog(@"retain count = %d", CFGetRetainCount(cfObject));    // 1
  NSArray * arr = CFBridgingRelease(cfObject);           // retainCount + 1 - 1
  NSLog(@"retain count = %d", CFGetRetainCount(cfObject));    // 1
  NSLog(@"%@", arr);
}

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

时间: 2024-10-25 21:34:50

id 与 void * 转换的相关文章

id 与void *类型的转换(转)

在ARC 无效时,像以下代码这样将id 变量强制转换void * 变量并不会出问题./* ARC 无效 */id obj = [[NSObject alloc] init];void *p = obj; 更进一步,将该void * 变量赋值给id 变量中,调用其实例方法,运行时也不会有问题./* ARC 无效 */id o = p;[o release]; 但是在ARC 有效时这便会引起编译错误.error: implicit conversion of an Objective-C point

iOS开发·runtime原理新葡京网站开发与实践: 基本知识篇

运行时新葡京网站开发haozbbs.comQ1446595067 1.1 基本概念: 运行时 Runtime 的概念 Runtime 又叫运行时,是一套底层的 C 语言 API,其为 iOS 内部的核心之一,我们平时编写的 OC 代码,底层都是基于它来实现的.比如: 以上你可能看不出它的价值,但是我们需要了解的是 Objective-C 是一门动态语言,它会将一些工作放在代码运行时才处理而并非编译时.也就是说,有很多类和成员变量在我们编译的时是不知道的,而在运行时,我们所编写的代码会转换成完整的

mybatis自定义枚举转换类

mybatis提供了 EnumTypeHandler和EnumOrdinalTypeHandler完成枚举类型的转换,两者的功能已经基本满足了日常的使用.但是可能有这 样的需求:由于某种原因,我们不想使用枚举的name和ordinal作为数据存储字段.mybatis的自定义转换类出现了. 示例 使用一段代码,将枚举类EnumStatus中的code属性存储到数据库对应字段statusCustom. 自定义转换类 package com.sg.util.typehandler; import ja

Spring MVC登录注册以及转换json数据

项目结构; 代码如下: BookController package com.mstf.controller; import javax.servlet.http.HttpServletResponse; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.codehaus.jackson.map.ObjectMapper; import com.mstf.

利用JAXB实现java实体类和xml互相转换

1.应用场景 在使用WebService实现数据上传下载,数据查询时,可以利用JAXB实现java实体类和xml互相转换 2.Demo 2.1 student.java 实体类,包含list(set同理).map.Teacher.Date 类型的属性 package jaxb; import java.util.Date; import java.util.List; import java.util.Map; import javax.xml.bind.annotation.XmlAccess

id类型

id类型 在Objective-C 中,id 类型是一个独特的数据类型.在概念上,类似Java 的Object 类,可以转换为任何数据类型.换句话说,id 类型的变量可以存放任何数据类型的对象.在内部处理上, 这种类型被定义为指向对象的指针,实际上是一个指向这种对象的实例变量的指针. 例如,下面定义了一个id类型的变量和返回一个id类型的方法: id anObject; - (id) newObject: (int) type; id 和void *并非完全一样.下面是id在objc.h中的定义

非ARC项目转换成ARC项目的相关支持

1.将项目编译环境改成arc 将红圈处从No 改成Yes 如果你现在的工程不支持ARC技术,你可以通过一个自动转换工具来转换你的工程(工具在Edit>Refactor>Convertto Objective-C ARC),这个工具会自动所有工程中手动管理内存的点转换成合适自动方式的(比如移除retain, release等).这个工具会转换工程中所有的文件.当然你可以转换单个文件. 2.下面的这些函数:dealloc,retain, release, retainCount, autorele

json格式转换

json数据转换 import java.io.IOException; import java.io.StringWriter; import java.util.ArrayList; import java.util.Collections; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import org.apache.commons.lang3.StringUtils; impo

【第3篇】通过JSON-Lib把数组转换成Json数据

package ivyy.taobao.com.domain.jsonlib; import ivyy.taobao.com.entity.Address; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import net.sf.json.JSONArray; import net.sf.json.JSONObject; /** *@DEMO:j