浅谈runtime关联

作者: 强子ly | 来源:发表于2017-11-14 19:44 被阅读654次

目录

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

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] 强子-男-努力到无能为力,拼搏到感动自己

相关文章

  • 浅谈runtime关联

    浅谈runtime关联 浅谈runtime关联

  • 浅谈runtime关联

    目录 序:面试中并不绝对的答案 开篇 核心代码 实例二则 1、序:面试中并不绝对的答案     今儿听到一个哥们讲...

  • Runtime总结

    参考: Objc Runtime 总结runtime 一, runtime 关联属性 1,设置关联值 object...

  • Runtime 关联对象, 可在分类中添加属性

    Runtime 关联对象, 可在分类中添加属性 关联 API 如下 设置关联值 获取关联值 取消关联 关联策略

  • [iOS开发]--Runtime的简单使用之关联对象

    一、Runtime关联对象的方法简介: 在中,有三个关联的方法,分别是: 1.1、...

  • NSRuntime使用篇

    使用总结runtime开源代码 对象的关联: 设置关联值 void objc_setAssociatedObjec...

  • ios运行时一些方法说明

    一. 关联对象主要函数: 基本说明:关联对象就是runTime界的NSMultableDictionary voi...

  • runtime关联属性

    前言 在开发中经常需要给已有的类添加方法和属性,但是Objective-C是不允许给已有类通过分类添加属性的,因为...

  • runtime - 关联属性

    我们知道苹果不允许我们自己给已经存在的类通过分类添加方法的,但是有时候我们确实需要给某个类从而分类添加属性,那么我...

  • runtime关联属性

    类别是开发中常常使用的东西,但是偶尔需要添加属性,这个就需要使用继承来实现,但是如果是只有一两个属性的话,也太过麻...

网友评论

    本文标题:浅谈runtime关联

    本文链接:https://www.haomeiwen.com/subject/pthfvxtx.html