浅谈runtime关联

目录

  • 序:面试中并不绝对的答案
  • 开篇
  • 核心代码
  • 实例二则

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 完全解析

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 <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] 强子-男-努力到无能为力,拼搏到感动自己
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,199评论 30 471
  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 3,732评论 7 64
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,837评论 18 139
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,757评论 0 9
  • 上午休息,下午会议培训。 早上起来去吃早餐。太多选择以致不知道该怎么搭配,中餐西餐胡乱混搭。忽然觉得人生好像就和这...
    热爱运动的小雨儿阅读 248评论 0 0