关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分。
- 关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的。
- 使用关联,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。
- 关联是基于关键字的。因此,我们可以为任何对象增加任意多的关联,每个都使用不同的关键字即可。
- 关联是可以保证被关联的对象在关联对象的整个生命周期都是可用的(在垃圾自动回收环境下也不会导致资源不可回收)。
- 使用关联时会涉及到3个OC的运行时(runtime)函数:objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects
创建关联
- 创建关联要使用objc_setAssociatedObject函数来把一个对象与另外一个对象进行关联。该函数需要四个参数:源对象,关键字,关联的对象和一个关联策略。
- 关键字是一个void类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。
- 关联策略表明了相关的对象是通过赋值、保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。
下面展示如何把一个字符串关联到一个数组上。
static char overviewKey;
NSArray *array = [[NSArray alloc] initWithObjects:@"One", @"Two", @"Three", nil];
// 为了演示,这里使用initWithFormat:来确保字符串可以被销毁
NSString *overview = [[NSString alloc] initWithFormat:@"numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
// ①此时overview仍然可用
NSLog(@"%@", overview);
[array release];
// ②此时overview不可用
- 在①处,字符串overview仍然是可用的,这是因为OBJC_ASSOCIATION_RETAIN策略指明了数组要保有相关的对象。关联后,字符串的引用计数为2。
- 在②处,当数组array被销毁的时候,overview也就会被释放,其引用计数为0,因此被销毁。如果此时还想使用overview,例如输出overview的值,则会出现运行时异常。
获取被关联的对象
获取被关联的对象时使用objc_getAssociatedObject函数:
NSString *associatedObject = objc_getAssociatedObject(array, &overviewKey);
NSLog(@"associatedObject = %@", associatedObject);
断开关联
断开关联是使用objc_setAssociatedObject函数,传入nil值即可:
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
- 其中,被关联的对象为nil,此时关联策略也就无关紧要了。
- 使用函数objc_removeAssociatedObjects可以断开所有关联。通常情况下不建议使用这个函数,因为它会断开所有关联。只有在需要把对象恢复到“原始状态”的时候才会使用这个函数。
下面是完整demo:
#import <Foundation/Foundation.h>
#import <objc/runtime.h>
int main(int argc, const char * argv[])
{
static char overviewKey;
NSArray *array = [[NSArray alloc] initWithObjects:@"One", @"Two", @"Three", nil];
// 为了演示,这里使用initWithFormat:来确保字符串可以被销毁
NSString *overview = [[NSString alloc] initWithFormat:@"numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
// 此时overview仍然可用
NSLog(@"%@", overview);
NSString *associatedObject = objc_getAssociatedObject(array, &overviewKey);
NSLog(@"associatedObject = %@", associatedObject);
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
NSString *newObject = objc_getAssociatedObject(array, &overviewKey);
// 输出为空
NSLog(@"newObject = %@", newObject);
return 0;
}