目录
序:面试中并不绝对的答案
开篇
核心代码
实例二则
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>, <#const void="">, <#id>, <#objc_associationpolicy>)
/** 获取关联对象 @param object#> 被关联对象 @param key#> 关键字(属性) */ objc_getAssociatedObject(<#id>, <#const void="">)
/* * 断开关联:直接传nil即可 */ objc_setAssociatedObject(array, &associatedKey, nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
3.1)关于 OBJC_ASSOCIATION_RETAIN_NONATOMIC,点进去我们发现这是一个枚举。对应属性修饰符见下表。表格转自关联对象 AssociatedObject 完全解析
runtime.png
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 @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 @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] 强子-男-努力到无能为力,拼搏到感动自己
作者:强子ly