1.概念
objective-c
有两个扩展机制:category
和associative
。我们可以通过category
来扩展方法,但是它有个很大的局限性,不能扩展属性。于是,就有了专门用来扩展属性的机制:associative
。
2.使用方法
在iOS
开发过程中,category
比较常见,而associative
就用的比较少。associative
的主要原理,就是把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分。
使用associative
,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。
associative
是基于关键字的。因此,我们可以为任何对象增加任意多的associative
,每个都使用不同的关键字即可。associative
是可以保证被关联的对象在关联对象的整个生命周期都是可用的。
associative机制提供了三个方法:
objc_setAssociatedObject(idobject,constvoid*key,id value,objc_AssociationPolicypolicy)
objc_getAssociatedObject(idobject,constvoid*key)
objc_removeAssociatedObjects(idobject)
2.1创建associative
创建associative
使用的是:objc_setAssociatedObject
。它把一个对象与另外一个对象进行关联。该函数需要四个参数:源对象,关键字,关联的对象、关联策略。
关键字是一个void
类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。
关联策略表明了相关的对象是通过赋值,保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。
比如,我们想对一个UIView
,添加一个NSString
类型的tag
。可以这么做:
-(void)setTagString:(NSString*)value{
objc_setAssociatedObject(self,KEY_TAGSTRING, value,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
2.2.获取associative对象
获取相关联的是函数objc_getAssociatedObject
。
继续上面的例子,从一个UIView
的实例中,获取一个NSString
类型的tag
-(NSString*)tagString{
NSObject*obj= objc_getAssociatedObject(self,KEY_TAGSTRING);
if (obj &&[obj isKindOfClass:[NSStringclass]]){
return(NSString*)obj;
}
returnnil;
}
2.3.断开associative
断开associative
是使用objc_setAssociatedObject
函数,传入nil
值即可。
objc_setAssociatedObject(self,KEY_TAGSTRING, nil,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
使用函数objc_removeAssociatedObjects
也可以断开所有associative
。通常情况下不建议这么做,因为他会断开所有关联。
3. 应用场景
3.1.TagString
上面的例子提到,在UIView
中添加NSString
类型的标记,就是一个非常实用的例子。全部的代码如下:
@interface UIView(BDTag)
@property (nonatomic,retain)NSString*tagString;
-(UIView*)viewWithTagString:(NSString*)value;
@end
#import"UIView+BDTag.h"
#undef KEY_TAGSTRING
#defineKEY_TAGSTRING "UIView.tagString"
@implementation UIView(BDTag)
@dynamic tagString;
-(NSString*)tagString{
NSObject*obj= objc_getAssociatedObject(self,KEY_TAGSTRING);
if (obj &&[obj isKindOfClass:[NSStringclass]]){
return(NSString*)obj;
}
returnnil;
}
-(void)setTagString:(NSString*)value{
objc_setAssociatedObject(self,KEY_TAGSTRING, value,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(UIView*)viewWithTagString:(NSString*)value{
if (nil==value) {
returnnil;
}
for (UIView*subviewin self.subviews){
NSString*tag =subview.tagString;
if ([tagisEqualToString:value])
{
returnsubview;
}
}
returnnil;
}
@end
苹果虽然有提供NSInteger
类型的tag
属性,用于标记相应的UI
。但是在处理比较复杂的逻辑的时候,往往NSInteger
类型的标记不能满足需求。为其添加了NSString
类型的标记后。就能使用字符串,快速的标记UI
,并且使用viewWithTagString
方法,快速找到你所需要的UI
。
3.2.为NSObject子类添加任何信息
这是一个方便,强大,并且简单的类。利用associative
机制,为任何Object
,添加你所需要的信息。比如用户登录,向服务端发送用户名/密码时,可以将这些信息绑定在请求的项之中。等请求完成后,再取出你所需要的信息,进行逻辑处理。而不需要另外设置成员,保存这些数据。
具体的实现如下:
@interface NSObject(BDAssociation)
-(id)associatedObjectForKey:(NSString*)key;
-(void)setAssociatedObject:(id)objectforKey:(NSString*)key;
@end
#import
#import "NSObject+BDAssociation.h"
@implementation NSObject(BDAssociation)
static char associatedObjectsKey;
-(id)associatedObjectForKey:(NSString*)key{
NSMutableDictionary*dict= objc_getAssociatedObject(self,&associatedObjectsKey);
return[dictobjectForKey:key];
}
-(void)setAssociatedObject:(id)objectforKey:(NSString*)key {
NSMutableDictionary*dict= objc_getAssociatedObject(self,&associatedObjectsKey);
if (!dict){
dict = [[NSMutableDictionaryalloc]init];
objc_setAssociatedObject(self,&associatedObjectsKey,dict, OBJC_ASSOCIATION_RETAIN);
}
[dict setObject:objectforKey:key];
}
@end
3.3.内存回收检测
记得在我刚开始学iOS
开发的时候,经常出现内存泄露的问题,于是就在各个viewcontroller
的dealloc
中打Log
。这种方法虽然有效,但比较挫,不好管理。
这里贴出一种漂亮的解决方案,利用associative
机制。让object
在回收时,自动输出回收信息。
@interface NSObject(BDLogDealloc)
-(void)logOnDealloc;
@end
#import "NSObject+BDLogDealloc.h"
static char __logDeallocAssociatedKey__;
@interface LogDealloc: NSObject
@property (nonatomic,copy)NSString*message;
@end
@implementation NSObject(LogDealloc)
-(void)logOnDealloc{
if(objc_getAssociatedObject(self,&__logDeallocAssociatedKey__)== nil) {
LogDealloc* log =[[[LogDealloc alloc]init]autorelease];
log.message =NSStringFromClass(self.class);
objc_setAssociatedObject(self,&__logDeallocAssociatedKey__,log, OBJC_ASSOCIATION_RETAIN);
}
}
@end
@implementation LogDealloc
-(void)dealloc{
NSLog(@"dealloc:%@",self.message);
[_messagerelease];
[superdealloc];
}
@end
总结
以上便是几种associative
机制的使用例子。这只是强大的associative
功能中,小小的几个缩影。有了associative
,就能用简单的几行代码,解决曾经困扰我们许久的问题。