美文网首页
associative(关联)-利用Runtime扩展类的属性

associative(关联)-利用Runtime扩展类的属性

作者: 天蓬大元 | 来源:发表于2017-08-04 10:40 被阅读0次


基本概念

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


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

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

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


associative机制提供了三个方法:

OBJC_EXPORT

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

OBJC_EXPORT 

id objc_getAssociatedObject(id object, const void *key)

OBJC_EXPORT 

void objc_removeAssociatedObjects(id object)


创建associative

       创建associative使用的是:objc_setAssociatedObject。它把一个对象与另外一个对象进行关联。该函数需要四个参数:源对象,关键字,关联的对象、关联策略。

       关键字是一个void类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。

       关联策略表明了相关的对象是通过赋值,保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。


获取associative对象

        获取相关联的是函数objc_getAssociatedObject。


断开associative

        断开associative是使用objc_setAssociatedObject函数,关注的对象传入nil值即可。

        使用函数objc_removeAssociatedObjects可以断开所有associative。通常情况下不建议这么做,因为他会断开所有关联。


应用场景

==============在UIView中添加NSString类型的标记===============

       苹果虽然有提供NSInteger类型的tag属性,用于标记相应的ui。但是在处理比较复杂的逻辑的时候,往往NSInteger类型的标记不能满足需求。为其添加了NSString类型的标记后。就能使用字符串,快速的标记ui,并且使用viewWithTagString方法,快速找到你所需要的ui。

@interface UIView(BDTag)

@property (nonatomic, retain) NSString *tagString;

- (UIView *)viewWithTagString:(NSString *)value;

@end

#import "UIView+BDTag.h"

#undef   KEY_TAGSTRING

#define KEY_TAGSTRING     "UIView.tagString"

@implementation UIView(BDTag)

@dynamic tagString;

//属性的get方法

- (NSString *)tagString {

          NSObject *obj = objc_getAssociatedObject(self, KEY_TAGSTRING);

           if (obj && [obj isKindOfClass:[NSString class]]) {

                      return (NSString *)obj;

            }

            return nil;

}

//属性的set方法

- (void)setTagString:(NSString *)value {

             objc_setAssociatedObject(self, 

                                                        KEY_TAGSTRING,

                                                        value,

                                                        OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

//获取到tag对应的view

- (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

==================为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);

        //利用key获取到字典中对应的value

        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);

            }

            //将value存储到关联对象字典中

            [dict setObject:object forKey:key];

}

@end

=======================内存回收检测========================

       如果你刚接触iOS开发,那么内存泄露则是你怎么也绕不开的一道坎,当你遇到内存泄漏时,怎么办呢?在各个view controller的dealloc中打Log。这种方法虽然有效,但不好管理。

       利用associative机制。让object在回收时,自动输出回收信息。

@interface NSObject (BDLogDealloc)

- (void)logOnDealloc;

@end

#import "NSObject+BDLogDealloc.h"

#import <objc/runtime.h>

static char __logDeallocAssociatedKey__;

@interface LogDealloc : NSObject

@property (nonatomic, copy) NSString* message;

@end

@implementation NSObject (BDLogDealloc)

- (void)logOnDealloc {

         if(objc_getAssociatedObject(self, &__logDeallocAssociatedKey__) == nil) {

                  LogDealloc* log = [[LogDealloc alloc] init] ;

                  log.message = NSStringFromClass(self.class);

                  objc_setAssociatedObject(self,

                                                            &__logDeallocAssociatedKey__,

                                                            log,

                                                             OBJC_ASSOCIATION_RETAIN);

              }

}

@end

@implementation LogDealloc

- (void)dealloc {

                  NSLog(@"dealloc: %@", self.message);

}

@end

感谢网友http://www.cnblogs.com/mkai/p/6594668.html的文章。

相关文章

网友评论

      本文标题:associative(关联)-利用Runtime扩展类的属性

      本文链接:https://www.haomeiwen.com/subject/kkhvlxtx.html