iOS底层原理

objc_setAssociatedObject 关联详解

2018-12-04  本文已影响0人  翰墨_42a6
1.关联
2.关联本质-在类的定义之外为类增加额外的存储空间

使用关联,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。
关联是基于关键字的,因此,我们可以为任何对象增加任意多的关联,每个都使用不同的关键字即可。关联是可以保证被关联的对象在关联对象的整个生命周期都是可用的(在垃圾自动回收环境下也不会导致资源不可回收)。

3.创建关联(原理)
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)

 id object                     :表示关联者,是一个对象,变量名理所当然也是object
 const void *key               :获取被关联者的索引key
 id value                      :被关联者,这里是一个block
 objc_AssociationPolicy policy :关联时采用的协议,有assign,retain,copy等协议,一般使用OBJC_ASSOCIATION_RETAIN_NONATOMIC

  • 关键字 : 是一个void类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。
  • 关联策略表明了相关的对象是通过赋值,保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。
4.创建关联(步骤1)
//1.声明一个静态变量,用作关键字.此处也可以直接用一个字符串(但不推荐)
static char overviewKey;  

//2.创建一个数组变量
NSArray * array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];  

//3.创建一个字符串变量
//为了演示的目的,这里使用initWithFormat:来确保字符串可以被销毁  
NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];  

//4.关联代码操作(关键)
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);  

[overview release];  //(mrc中使用,arc中不用写)为了说明overview是否可用而写,ARC中省略
//(1) overview仍然是可用的  

[array release];  //(mrc中使用,arc中不用写)为了说明overview是否可用而写,ARC中省略
//(2)overview 不可用  

  • 说明:
    在(1)处,字符串overview仍然是可用的,这是因OBJC_ASSOCIATION_RETAIN
    策略指明了数组要保有相关的对象。当数组array被销毁的时候,也就是在(2)处overview也就会被释放,因此而被销毁。如果此时还想使用overview,例如想通过log来输出overview的值,则会出现运行时异常。
5.取出关联的对象去使用(步骤2)
NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &oveviewKey);  
//取出后,可以使用,打印,赋值等操作

6.断开关联
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);  

说明:
其中,被关联的对象为nil,此时关联策略也就无关紧要了。
使用函数objc_removeAssociatedObjects可以断开所有关联。通常情况下不建议使用这个函数,因为他会断开所有关联。只有在需要把对象恢复到“原始状态”的时候才会使用这个函数。

7.完整代码展示
//1.声明一个静态变量,用作关键字.此处也可以直接用一个字符串(但不推荐)
    static char overviewKey;

//2.创建一个数组变量
    NSArray *array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];  

//3.创建一个字符串变量
//为了演示的目的,这里使用initWithFormat:来确保字符串可以被销毁 
    NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];  

//4.关联代码操作(关键)
    objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);  

//5.使用的地方取出关联的对象
    NSString *associatedObject = (NSString *)objc_getAssociatedObject(arrray, &overviewKey);  

//6.使用--打印
    NSLog(@"associatedObject:%@", associatedObject);  

//7.断开关联      
    objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);  
    [array release];  (mrc中使用,arc中不用写)


参考文章

上一篇下一篇

猜你喜欢

热点阅读