【原】objc_setAssociatedObject和objc_getAssociatedObject
本文转载请注明出处——polobymulberry-博客园
两个函数名称中都有associate,意思是关联,这里的关联表示的是一种 从属关系,即有一个关联者和被关联者,我们说NSArray的对象array关联了NSString对象string,这里的array就是关联者(表示主动关联别人),string就是被关联者(表示被动被别人关联)。我们就会产生三个哲学问题。
- 关联是什么?
- 为什么关联?
- 怎么关联?
既然关联是要有关联者和被关联者,我们可以看到objc_setAssociatedObject这个函数,就是为了设定关联关系,说简单点,就是为了把两个对象关联起来。
// objc_setAssociatedObject函数原型
OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
- OBJC_EXPORT 打包lib时,用来说明该函数是暴露给外界调用的。
主要看该函数的几个参数:
- id object 表示关联者,是一个对象,变量名理所当然也是object
- id value 表示被关联者,我们可以看到它的变量名是value,我们这里一定要理解
这个value最后是要关联到object上的。
关联者和被关联者有了,那么又会产生两个问题,而答案正好对应函数的另外两个参数。
- 一个就是我
日后如何获取这个关联关系,也就是说,我今后要使用这个关联关系,如何获取?有人会说我直接使用根据object(关联者)来获取不就行了,但是一个object可能有许多被关联者,比如上面的NSArray可以关联一个NSString,也可以关联一个NSDictionary。有人说那直接用关联者和被关联者两个去获取不就行了。但是使用一个关键词key(一般是一个字符串),更合适。因为字符串可以写成全局的,这样可以在各个地方都自由获取到关联的对象了。而如果直接使用关联的对象来索引,并且当这个关联对象是局部变量时,那么在别的函数中要获取的时候,就很麻烦了,此时你要不然就写成全局的,那样代码将很乱,所以我觉得统一使用字符串来索引关联关系更合适。我们看看apple是怎么设计获取关联的方法:
OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)
你看,只需要一个object(关联者)和一个key来获取关联。这里我主观臆测此处一个关联者的关联关系使用hashtable来表示的(我猜测的,还未深入研究,大神请拍砖)。根据objc_getAssociatedObject的参数,我们理所当然就会在set函数中使用一个key来表示关联关系了。大概就像下面这样:

- 另外,这个被关联的对象也是要存在内存中的,那它的内存管理方式如何?
objc_setAssociatedObject函数中还有一个参数是
objc_AssociationPolicy policy
它是一个枚举,具体如下
enum {
OBJC_ASSOCIATION_ASSIGN = 0,
OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,
OBJC_ASSOCIATION_COPY_NONATOMIC = 3,
OBJC_ASSOCIATION_RETAIN = 01401,
OBJC_ASSOCIATION_COPY = 01403
};
这个字面上看是指关联策略,但是这里的策略更注重关联关系的内存管理方面。我们这里可以大胆意淫,这个被关联者value其实很像object的一个属性(property),当然,我们还可以扩展方法。看到这里,有人可能会说这不就是category吗?可以说是,也可以说不是。是的原因是它们都扩展了OC对象的属性,不是的原因是category是编译时就决定了扩展的属性,而objc_setAssociatedObject是运行时来扩展属性。
比如我们使用
objc_setAssociatedObject(array, &key, string, OBJC_ASSOCIATION_RETAIN);
相当于string是array的一个retain属性,嗯。。。你可以理解为strong。所以当array销毁时,string也自然就不存在了。
上面看懂了,使用就很方便了。举个最简单的例子,把一个NSString对象关联到NSArray对象上。
#import "ViewController.h"
// 使用objc_getAssociatedObject和objc_setAssociatedObject
// 需要添加objective-c的运行时文件
#import <objc/runtime.h>
// 表示关联关系的key,主要目的是用来索引
const NSString *associatedKey = @"associate_nsarray_with_nsstring_key";
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSArray *array = [NSArray arrayWithObjects:@"hello", @"world", @"!", nil];
NSString *string = @"I am an iOS developer!";
// 将string关联到array上
objc_setAssociatedObject(array, &associatedKey, string, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
// 从array中获取被关联的对象string
// 注意,这里就没有string这个对象任何事了
// string其实已经变成了array的一个属性值
NSString *getAssociatedObject = objc_getAssociatedObject(array, &associatedKey);
NSLog(@"%@", getAssociatedObject);
}
@end
输出结果

当然,oc中关联的方法可以做很多事,网上一搜一大把。我觉得首先明白这两个函数是做什么的,至于怎么用,什么最佳实践,还是去实践中体会了。
【原】objc_setAssociatedObject和objc_getAssociatedObject的更多相关文章
- [Objective-C]关联(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects)
关联 关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分. 关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的. 在类的定义之外为类增加额外的存储空间 ...
- [Objective-C]关联(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects)(转)
转载自:http://blog.csdn.net/onlyou930/article/details/9299169 分类: Objective-C2013-07-11 11:54 3420人阅读 评 ...
- ios扩展机制objc_setAssociatedObject,objc_getAssociatedObject
这个可以解决变量传递问题, 就不用定义全局的了. 使用例子: 首先导入头文件:#import <objc/runtime.h> 设置静态常量:static char alertinfoke ...
- iOS swift objc_setAssociatedObject和objc_getAssociatedObject使用
oc中的AssociationsManager在swift中也是可以实现的 使用方法请看下面一个例子 import UIKit extension UIButton { func fk_addActi ...
- iOS - Objective-C 关联(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects)
关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分. 关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的. 在类的定义之外为类增加额外的存储空间 使用关联,我 ...
- objc_setAssociatedObject 1
[Objective-C]关联(objc_setAssociatedObject.objc_getAssociatedObject.objc_removeAssociatedObjects) 标签: ...
- iOS Objective-C对象模型及应用
前言 原创文章,转载请注明出自唐巧的技术博客. 本文主要介绍Objective-C对象模型的实现细节,以及Objective-C语言对象模型中对isa swizzling和method swizzli ...
- iOS 消息发送与转发详解
Objective-C 是一门动态语言,它将很多静态语言在编译和链接时期做的事情,放到了运行时来处理.之所以能具备这种特性,离不开 Runtime 这个库.Runtime 很好的解决了如何在运行时期找 ...
- iOS Class结构分析
objc_class结构体 类在OC中是objc_class的结构体指针 typedef struct objc_class *Class; 在objc/runtime.h中objc_class结构体 ...
随机推荐
- session实现购物车
为实现简单的购物功能(购物车添加.账户查看.购物车商品删除.实时的购物商品数量及价格的计算显示.购物车商品数量可手动输入等),用session实现了一简单的以php语言为基础.连接MySQL数据库的购 ...
- 阿里云直播 C# SDK 如何使用
阿里云直播SDK的坑 1.直播云没有单独的SDK,直播部分被封装在CDN的相关SDK当中. 2.针对SDK,没有相关Demo. 3.针对SDK,没有相关的文档说明. 4.针对SDK的说明,官网上的说明 ...
- JS 判断数据类型的三种方法
说到数据类型,我们先理一下JavaScript中常见的几种数据类型: 基本类型:string,number,boolean 特殊类型:undefined,null 引用类型:Object,Functi ...
- ASP.NET Aries 入门开发教程9:业务表单的开发
前言: 经过前面那么多篇的列表的介绍,终于到了大伙期待的表单开发了. 也是本系列的最后一篇文章了! 1:表单页面的权限设置与继承 对于表单页面,权限的设置有两种: 1:你可以选择添加菜单(设置为不显示 ...
- Hello Web API系列教程——Web API与国际化
软件国际化是在软件设计和文档开发过程中,使得功能和代码设计能处理多种语言和文化习俗,在创建不同语言版本时,不需要重新设计源程序代码的软件工程方法.这在很多成熟的软件开发平台中非常常见.对于.net开发 ...
- app开发外包注意事项,2017最新资讯
我们见过很多创业者,栽在这app外包上.很多创业者对于app外包这件事情不是特别重视,以为将事情交给app外包公司就完事了,实际上不是的.无论是从选择app外包公司还是签订合同.售后维护等各方面都有许 ...
- JS里面Data日期格式转换
var format = function(time, format){ var t = new Date(time); var tf = function(i){return (i ...
- ASP.NET从零开始学习EF的增删改查
ASP.NET从零开始学习EF的增删改查 最近辞职了,但是离真正的离职还有一段时间,趁着这段空档期,总想着写些东西,想来想去,也不是很明确到底想写个啥,但是闲着也是够 ...
- UWP开发之Mvvmlight实践九:基于MVVM的项目架构分享
在前几章介绍了不少MVVM以及Mvvmlight实例,那实际企业开发中将以那种架构开发比较好?怎样分层开发才能节省成本? 本文特别分享实际企业项目开发中使用过的项目架构,欢迎参照使用!有不好的地方欢迎 ...
- python笔记(持续更新)
1.编译python遇到下面的编码问题: SyntaxError: Non-ASCII character '\xe9' in file E:\projects\learn.py on lin ...