Associative机制的学习总结

出处:http://blog.sina.com.cn/s/blog_60342e330101tcz1.html


1.    概念

objective-c有两个扩展机制:categoryassociative。我们可以通过category来扩展方法,但是它有个很大的局限性,不能扩展属性。于是,就有了专门用来扩展属性的机制:associative

 

2.    使用方法

iOS开发过程中,category比较常见,而associative就用的比较少。associative的主要原理,就是把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分。

使用associative,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。

associative是基于关键字的。因此,我们可以为任何对象增加任意多的associative,每个都使用不同的关键字即可。associative是可以保证被关联的对象在关联对象的整个生命周期都是可用的。

associative机制提供了三个方法:

   

OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value,objc_AssociationPolicy policy)

 

OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)

 

OBJC_EXPORT void objc_removeAssociatedObjects(id object)

 

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:[NSString class]]) {
              return (NSString *)obj;
    }
        
         return nil;
}

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
</pre><pre name="code" class="objc">#import "UIView+BDTag.h"
 
#undef   KEY_TAGSTRING
#define KEY_TAGSTRING     "UIView.tagString"
 
@implementation UIView(BDTag)
 
@dynamic tagString;
 
- (NSString *)tagString {
         NSObject *obj = objc_getAssociatedObject(self, KEY_TAGSTRING);
         if (obj && [obj isKindOfClass:[NSString class]]) {
              return (NSString *)obj;
    }
        
         return nil;
}
 
- (void)setTagString:(NSString *)value {
         objc_setAssociatedObject(self, KEY_TAGSTRING, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
 
- (UIView *)viewWithTagString:(NSString *)value {
         if (nil == value) {
              return nil;
    }
        
         for (UIView *subview in self.subviews) {
              NSString *tag = subview.tagString;
              if ([tag isEqualToString:value])
              {
                     return subview;
              }
         }
        
         return nil;
}
 
@end
苹果虽然有提供 NSInteger 类型的 tag 属性,用于标记相应的 ui 。但是在处理比较复杂的逻辑的时候,往往 NSInteger 类型的标记不能满足需求。为其添加了 NSString 类型的标记后。就能使用字符串,快速的标记 ui ,并且使用 viewWithTagString 方法,快速找到你所需要的 ui


3.2.NSObject子类添加任何信息

这是一个方便,强大,并且简单的类。利用associative机制,为任何Object,添加你所需要的信息。比如用户登录,向服务端发送用户名/密码时,可以将这些信息绑定在请求的项之中。等请求完成后,再取出你所需要的信息,进行逻辑处理。而不需要另外设置成员,保存这些数据。

具体的实现如下:

@interface NSObject (BDAssociation)
 
- (id)associatedObjectForKey:(NSString*)key;
- (void)setAssociatedObject:(id)object forKey:(NSString*)key;
 
@end
 
#import "NSObject+BDAssociation.h"
 
@implementation NSObject (BDAssociation)
 
static char associatedObjectsKey;
 
- (id)associatedObjectForKey:(NSString*)key {
    NSMutableDictionary *dict = objc_getAssociatedObject(self, &associatedObjectsKey);
    return [dict objectForKey:key];
}
 
- (void)setAssociatedObject:(id)object forKey:(NSString*)key {
    NSMutableDictionary *dict = objc_getAssociatedObject(self, &associatedObjectsKey);
    if (!dict) {
        dict = [[NSMutableDictionary alloc] init];
        objc_setAssociatedObject(self, &associatedObjectsKey, dict, OBJC_ASSOCIATION_RETAIN);
    }
    [dict setObject:object forKey:key];
}
 
@end

3.3.内存回收检测

记得在我刚开始学iOS开发的时候,经常出现内存泄露的问题,于是就在各个view controllerdealloc中打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);
    [_message release];
    [super dealloc];
}
 
@end

4.   总结

以上便是几种associative机制的使用例子。这只是强大的associative功能中,小小的几个缩影。有了associative,就能用简单的几行代码,解决曾经困扰我们许久的问题。




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值