本文是关于Category关联对象的一些理解化,以及部分使用建议。仅适用于新手玩家(对Category关联对象停留在只会使用的阶段)。
关联对象的使用
@interface JHObject : NSObject
@property (nonatomic,strong) NSString *myProperty;
@end
在OC中我是创建属性一般使用property。关于property我们可以理解为OC的一个宏定义。它会让编译器帮我们自动生成实例变量和Get方法和Set方法
property = 实例变量 + getter + setter
但是,在分类中仅使用property添加属性是无效的,这个时候编译器也会警告我们无法自动帮我们生成实例变量已经getter和setter。这是由于分类是动态加载的,在加载分类之前我们原本的对象已经加载到内存当中,如果我们这个时候加入对象,就会影响之前的内存结构,导致一些不可测的问题。
上面解析了为什么分类中直接使用property无法直接添加属性。下面介绍的是分类关联对象的使用,使我们可以在分类中正常添加属性。
@interface JHObject(category)
@property (nonatomic,strong) NSString *categoryProperty;
@end
@implementation JHObject(category)
-(NSString*)categoryProperty{
return objc_getAssociatedObject(self, _cmd);
}
- (void)setCategoryProperty:(NSString *)categoryProperty {
objc_setAssociatedObject(self, @selector(categoryProperty), categoryProperty, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end
一般我们都是使用 objc_getAssociatedObject 以及 objc_setAssociatedObject 来模拟属性的存取方法,使用关联对象模拟实例变量。
关于方法属性的解析
1. _cmd 方法的隐藏属性,代表当前方法的选择子,也就是 @selector(categoryProperty)。
id objc_getAssociatedObject(id object, const void *key);
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);
上面是两个方法的原型,我们可以看到_cmd 代替的参数是key。它的类型是 static void * 。在这之前本人也是手动注册字符串来传入。后来参考了别人的做法,我们可以使用@selector(categoryProperty)作为key传入。这个做法不仅保证了keuy的唯一性,同时也免去了手动注册字符串的麻烦。
2. OBJC_ASSOCIATION_RETAIN_NONATOMIC 代表的是我们在property中使用的修饰符。
objc_AssociationPolicy | modifier |
---|---|
OBJC_ASSOCIATION_ASSIGN | assign |
OBJC_ASSOCIATION_RETAIN_NONATOMIC | nonatomic, strong |
OBJC_ASSOCIATION_COPY_NONATOMIC | nonatomic, copy |
OBJC_ASSOCIATION_RETAIN | atomic, strong |
OBJC_ASSOCIATION_COPY | atomic, copy |
以上是关联属性的一些使用方法已经使用建议
使用特例
上面例子添加的属性是NSString 是属于NSObject的子类,但是当我们添加BOOL这种基本数据类型应该怎么办呢。下面就为分类添加一个BOOL类型的属性
@interface JHObject(categoty)
@property (nonatimoc,assign)BOOL loadSuccess;
@end
@implementation JHObject(category)
-(BOOL)loadSuccess{
return [objc_getAssociatedObject(self, loadSuccess_key) boolValue];
}
-(void)setLoadSuccess:(BOOL)loadSuccess{
objc_setAssociatedObject(self, loadSuccess_key, [NSNumber numberWithBool:loadSuccess], OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@nd
从上面的代码中我们就可以看到我们可以将BOOL当做一个对象进行处理,只要我们在存取的方法中进行转化就可以了。