在面试的时候,我们也许会遇到这样的面试题:在不使用继承的情况下,如何给一个类添加属性?其实给类添加属性还可以创建分类,利用iOS runtime ,关联对象的方法给类添加属性。
什么是关联对象?
Associated Objects(关联对象)或者叫作关联引用(Associative References),是作为Objective-C 2.0 运行时功能被引入到 Mac OS X 10.6 Snow Leopard(及iOS4)系统。与它相关在<objc/runtime.h>中有3个C函数,它们可以让对象在运行时关联任何值:
//设置关联对象
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)
其实简单点说就是把两个没有关系的东西,关联在一起。
设置关联对象(Set方法)
//设置关联对象
OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
这个方法有4个参数:
1.源对象;
2.关键字:这个 key 值必须保证是一个对象级别的唯一常量。有以下三种推荐的 key 值:
1.声明 static char kAssociatedObjectKey,使用 &kAssociatedObjectKey 作为 key 值;
2.声明 static void *kAssociatedObjectKey = &kAssociatedObjectKey ,使用 kAssociatedObjectKey 作为 key 值;
3.用 selector ,使用 getter 方法的名称作为 key 值。
3.关联的对象;
4.关联策略:有五种关联策略可供选择:
/**
* OBJC_ASSOCIATION_ASSIGN
等价于@property (assign) ,
@property (unsafe_unretained)
弱引用关联对象
* OBJC_ASSOCIATION_RETAIN_NONATOMIC
等价于@property (strong, nonatomic)
强引用关联对象,且为非原子操作
* OBJC_ASSOCIATION_COPY_NONATOMIC
等价于@property (copy, nonatomic)
复制关联对象,且为非原子操作
* OBJC_ASSOCIATION_RETAIN
等价于@property (strong, atomic)
强引用关联对象,且为原子操作
* OBJC_ASSOCIATION_COPY
等价于@property (copy, atomic)
复制关联对象,且为原子操作
其中,第 2 种与第 4 种、第 3 种与第 5 种关联策略的唯一差别就在于操作是否具有原子性。
*/
获取关联对象(get方法)
//获取关联对象
OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)
断开对象的关联
我们不需要主动调用removeAssociated来接触关联的对象,如果需要解除指定的对象,可以使用setAssociatedObject置nil来实现。
吃个🌰:
.h
#import <UIKit/UIKit.h>
@interface UIViewController (Alpha)
@property (nonatomic, copy) NSString *strNavBarBgAlpha;
@end
.m
@implementation UIViewController (Alpha)
- (void)setStrNavBarBgAlpha:(NSString *)strNavBarBgAlpha {
//设置关联对象
objc_setAssociatedObject(self, @selector(strNavBarBgAlpha), strNavBarBgAlpha, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (NSString *)strNavBarBgAlpha {
//获取关联对象
return objc_getAssociatedObject(self, @selector(strNavBarBgAlpha));
}
@end
调用:
self.strNavBarBgAlpha = @"1.0";