目录
- 序:面试中并不绝对的答案
- 开篇
- 核心代码
- 实例二则
1、序:面试中并不绝对的答案
今儿听到一个哥们讲,前几天面试,面试官问的几个问题刚好是自己背的滚瓜烂熟的,心中一阵窃喜,滔滔不绝,最后居然音信全无。殊不知,碰到大咖面试官,你只知道字面意思而不知其所以然,是很容让人看透你的。例如:
1.1)类别的作用?继承和类别在实现中有何区别?
category 可以在不获悉,不改变原来代码的情况下往里面添加新的方法,只能添加,不能删除修改,并且如果类别和原来类中的方法产生名称冲突,则类别将覆盖原来的方法,因为类别具有更高的优先级
1.2)类别和类扩展的区别
category和extensions的不同在于 后者可以添加属性。另外后者添加的方法是必须要实现的;可以认为是一个私有的Category。
1.3)序总结
这是我在 iOS开发笔试-100道 中整理的两道关于Category的面试题,看整理的答案,可以说并没有什么问题,但经过这哥们这么一面,我发现部分还是有瑕疵的。如果面试官想问你【runtime关联】,动态创建属性,而你一再强调Category不可添加属性,虽然各执一词,但人家现在是big brother。
怎么办呢?接下来这篇文章就是为你量身定做。
2、开篇
我们在 iOS 开发中经常需要使用分类(Category),为已经存在的类添加属性的需求,但是使用 @property 并不能在分类中正确创建实例变量和存取方法。
不过,通过 Objective-C 运行时中的关联对象,也就是Associated Object,我们可以实现上述需求。
目的:使用runtime实现Category增加属性
3、核心代码
objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);
objc_getAssociatedObject(id object, const void *key);
objc_removeAssociatedObjects(id object);
/**
关联
@param object#> 源对象
@param key#> 关键字
@param value#> 关联的对象
@param policy#> 关联策略
*/
objc_setAssociatedObject(<#id object#>,
<#const void *key#>,
<#id value#>,
<#objc_AssociationPolicy policy#>)
/**
获取关联对象
@param object#> 被关联对象
@param key#> 关键字(属性)
*/
objc_getAssociatedObject(<#id object#>, <#const void *key#>)
/*
* 断开关联:直接传nil即可
*/
objc_setAssociatedObject(array,
&associatedKey,
nil,
OBJC_ASSOCIATION_RETAIN_NONATOMIC)
3.1)关于 OBJC_ASSOCIATION_RETAIN_NONATOMIC,点进去我们发现这是一个枚举。对应属性修饰符见下表。表格转自关联对象 AssociatedObject 完全解析
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 |
4、实例二则
看了网上一个烂大街的例子,不知道是谁写的了,虽然被大量的复制,说不上经典,但讲的深入浅出,也可以说是非常容易理解了。
4.1)、把一个字符串关联到一个数组上
const NSString *associatedKey = @"associate_nsarray_with_nsstring_key";
NSArray *array = [NSArray arrayWithObjects:@"hello", @"world", nil];
NSString *string = @"I am an iOS developer!";
/*
* 将string关联到array上
*/
objc_setAssociatedObject(array,
&associatedKey,
string,
OBJC_ASSOCIATION_RETAIN_NONATOMIC);
/*
* 从array中获取被关联的对象string
* 注意,这里就没有string这个对象任何事了
* string其实已经变成了array的一个属性值
*/
NSString *getAssociatedObject = objc_getAssociatedObject(array,
&associatedKey);
NSLog(@"%@", getAssociatedObject);
4.2)、给Category增加属性。
创建一个简单的用户模型,包含“ 姓名”、“ 性别”,再创建一个用户模型的Category,在Category内增加一个“个性签名”属性。
用户模型
#import <Foundation/Foundation.h>
@interface UserModel : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *gender;
@end
理论上Category不能添加属性,只能添加
⚠️❌错误写法
#import "UserModel.h"
@interface UserModel (Cate)
@property (nonatomic, copy) NSString *remark;
@end
⚠️❌错误输出
UserModel *model = [[UserModel alloc] init];
model.name = @"强子";
model.gender = @"男";
model.remark = @"努力到无能为力,拼搏到感动自己";
NSLog(@"%@-%@-%@",model.name,model.gender,model.remark);
unrecognized selector sent to instance 0x60000003f4a0'
*** First throw call stack
✅runtime增加伪属性
#import "UserModel.h"
@interface UserModel (Cate)
- (NSString *)remark;
- (void)setRemark:(NSString *)remark;
@end
#import "UserModel+Cate.h"
#import <objc/runtime.h>
@implementation UserModel (Cate)
/*
* 使用关联对象模拟实例变量
* 使用objc_getAssociatedObject、objc_setAssociatedObject模拟『属性』的存取方法
*/
- (NSString *)remark
{
return objc_getAssociatedObject(self, _cmd);
}
- (void)setRemark:(NSString *)remark
{
objc_setAssociatedObject(self,
@selector(remark),
remark,
OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end
UserModel *model = [[UserModel alloc] init];
model.name = @"强子";
model.gender = @"男";
model.remark = @"努力到无能为力,拼搏到感动自己";
NSLog(@"%@-%@-%@",model.name,model.gender,model.remark);
2017-11-14 18:03:44.901 Runtime[7689:270941] 强子-男-努力到无能为力,拼搏到感动自己