runtime - 关联属性

我们知道苹果不允许我们自己给已经存在的类通过分类添加方法的,
但是有时候我们确实需要给某个类从而分类添加属性,那么我们该怎么办呢?
通过这一节学习,我们也许就会有思路了!

为了引出今天要学习的东西,假设我们的项目有这样一个需求,我们要给所有继承自NSObjct的类增加一个objName的属性,而且这个属性必须卸载分类里边,好了那么在我们还不是太了解runtime的情况下,我们可能会这么做:
新建一个NSObjct的分类,命名为ObjConnect,在.h文件里边,我们声明一个属性:

/** 我么知道如果我们给自定义的分类增加了某个属性是没问题的,但是如果是给某个分类增加一个属性那么会有什么问题呢?
 *  在这里我们只在NSObjct的分类里边增加一个objName的属性,而不做任何其他事情,那么我们看看会有什么问题 
 */
@property (nonatomic, copy) NSString *objName;

然后我们再新建一个Dog类,继承自NSObjct,除此之外我们不做任何其他事情。

在我们的控制器里边,初始化一个Dog对象,然后给objName初始化一个值并打印之,代码如下:

Dog *dog = [Dog new];
dog.objName = @"旺财";
NSLog(@"%@", dog.objName);

好了,让我们运行程序:

2016-07-05 14:44:06.699 ZFRuntime[1314:135570] -[Dog setObjName:]: unrecognized selector sent to instance 0x7fbed341c7b0
2016-07-05 14:44:06.703 ZFRuntime[1314:135570] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Dog setObjName:]: unrecognized selector sent to instance 0x7fbed341c7b0'

这时候我们看到程序崩溃了,并且控制台输出了如上的错误,提示[Dog setObjName:]这个方法系统并不认识,因为程序不允许我们给分类添加这样的属性。

下边我们在思考,既然提示[Dog setObjName:]这个方法认识,那么我们自己写objName的setget方法行不行呢?按照这个思路我们再来试一下,在NSObjct这个分类里边增如下几行代码:

/** 我们看看自己来写`objName`的`set`和`get`方法 */
- (void)setObjName:(NSString *)objName
{
    self.objName = objName;
}

- (NSString *)objName
{
    return self.objName;
}

然后我们再次运行程序:这个时候程序出现了一个更严重的问题,提示内存出现泄漏,看来这样也打不到我们的目的。

既然上边两个思路都行不通,那么让我们用运行时的方式看看能不能解决这个问题呢?

首先在NSObjct分类的.m文件里边定义一个关联值:

/** 系统将会通过这个关联值来获取属性值,注意必须保证这个值唯一,通常使用`static const void *`方式来声明 */
static const void *TagSetObjNameKey;

然后实现在两个方法:

/** 设置关联的值,也就是设置`objName`属性的值,可以理解为这个属性的`setter`方法 */
- (void)setObjName:(NSString *)objName
{
    objc_setAssociatedObject(self, TagSetObjNameKey, objName, OBJC_ASSOCIATION_COPY);
}

/** 获取关联的值,也就是获取`objName`,可以理解为这个属性的`getter`方法 */
- (NSString *)objName
{
    return objc_getAssociatedObject(self, TagSetObjNameKey);
}

好了让我们再次运行程序,这是口可以看到控制台打印信息:

2016-07-05 15:01:26.036 ZFRuntime[1424:145877] 旺财

说明我们的目的已经实现了,通过runtime的方法,我们成功地给分类增加了一个属性!

既然运行时的这个API这么方便,那么再来分析下这几个方法到底什么意思,我们以后要是使用的话应该传递什么参数呢?

苹果官方文档里边,和属性关联相关的API有三个,这三个API分别是:

/**
*  定义:通过给定关联的`类`和`关联策略`等,为某个类增加关联(属性)
*
*  @param object 要关联的(要增加属性的)那个类
*  @param key    关联的键,例如上边例子中我们传递的是 `TagSetObjNameKey`
*  @param value  通过`键`要关联的那个值,我们简单可以理解为我们自定义的这个属性。(如果要清除某个关联,这个值可以设置为nil,但是很少这么做)
*  @param policy 关联策略(关于这个参数下边我们会详解)
*/
void objc_setAssociatedObject(id object, void *key, id value, objc_AssociationPolicy policy)

/**
*  定义:通过给定`类`和`关联键`获取关联(属性)
*
*  @param object 同上
*  @param key    同上
*
*  @return 关联的值(属性)
*/
id objc_getAssociatedObject(id object, void *key)

/**
*  定义:给定关联的`类`,移除所有关联(这个方法)
*  备注:这个方法很少用
*  @param object 同上
*/
void objc_removeAssociatedObjects(id object)

这里有几点我们有必要说明一下:

  • 关于void objc_setAssociatedObjectobjc_getAssociatedObjectvoid *key这个参数,其实这个参数有很多种写法,比较常见的有以下几种:

    • 示例程序中的写法,即 static const void *关联键;,这中写法在SDWebImage中比较常见
    • _cmd的方式(_cmd代表本方法的名称),关于这个东西可以参考这里,笔者第一次见到这个用法是在这个demo里边
    • 还有一种方法是@selector(方法名)的方式
      其实这几种写法都大同小异,根据个人爱好来就可以了
  • 关于objc_setAssociatedObject方法中objc_AssociationPolicy policy这个参数,苹果官方是这样定义的,简单理解就是引用方式,相当于我们在声明属性是()里边写的那几个东西:

      enum {
          OBJC_ASSOCIATION_ASSIGN = 0, // (assign)
          OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, (retain, nonactomic)
          OBJC_ASSOCIATION_COPY_NONATOMIC = 3, (copy, nonactomic)
          OBJC_ASSOCIATION_RETAIN = 01401, (retain)
          OBJC_ASSOCIATION_COPY = 01403 (copy)
      };
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容