iOS 运行时之 Associative(关联)

原文地址:http://www.jkeabc.com/649648.html
附:关联对象 AssociatedObject 完全解析http://www.jianshu.com/p/79479a09a8c0

iOS 运行时之 Associative(关联)
iOS 下有很多运行时特性,这里介绍一下 Associative(关联) 这个运行时特性,以及它一些使用场景。 Associative 意思为关联,
iOS 下有很多运行时特性,这里介绍一下 Associative(关联)

这个运行时特性,以及它一些使用场景。 Associative

意思为关联,能够将两个对象建立一种关系。这种关系是一种 从属

关系,也就是说有一个 关联者

和一个 被关联者

。比如说我们可以将一个 NSString

对象关联到一个 UIView

对象上。这里的 NSString

对象就是 被关联者

, UIView

对象就是 关联者

在 objc/runtime.h

文件中,找到 Associative

相关的 API 定义,如下:

/** 
 * Sets an associated value for a given object using a given key and association policy.

 * 

 * @param object The source object for the association.

 * @param key The key for the association.

 * @param value The value to associate with the key key for object. Pass nil to clear an existing association.

 * @param policy The policy for the association. For possible values, see “Associative Object Behaviors.”

 * 

 * @see objc_setAssociatedObject

 * @see objc_removeAssociatedObjects

 */

OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)

    OBJC_AVAILABLE(10.6, 3.1, 9.0, 1.0);

/** 

 * Returns the value associated with a given object for a given key.

 * 

 * @param object The source object for the association.

 * @param key The key for the association.

 * 

 * @return The value associated with the key /e key for /e object.

 * 

 * @see objc_setAssociatedObject

 */

OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)

    OBJC_AVAILABLE(10.6, 3.1, 9.0, 1.0);

/** 

 * Removes all associations for a given object.

 * 

 * @param object An object that maintains associated objects.

 * 

 * @note The main purpose of this function is to make it easy to return an object 

 *  to a "pristine state”. You should not use this function for general removal of

 *  associations from objects, since it also removes associations that other clients

 *  may have added to the object. Typically you should use /c objc_setAssociatedObject 

 *  with a nil value to clear an association.

 * 

 * @see objc_setAssociatedObject

 * @see objc_getAssociatedObject

 */

同时还提供以下枚举类型的定义:

/**
 * Policies related to associative references.

 * These are options to objc_setAssociatedObject()

 */

typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy) {

    OBJC_ASSOCIATION_ASSIGN = 0,           /**< Specifies a weak reference to the associated object. */

    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, /**< Specifies a strong reference to the associated object. 

                                            *   The association is not made atomically. */

    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,   /**< Specifies that the associated object is copied. 

                                            *   The association is not made atomically. */

    OBJC_ASSOCIATION_RETAIN = 01401,       /**< Specifies a strong reference to the associated object.

                                            *   The association is made atomically. */

    OBJC_ASSOCIATION_COPY = 01403          /**< Specifies that the associated object is copied.

                                            *   The association is made atomically. */

};

API 解析
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)

API 为我们提供了将两个对象建立关联关系的能力,参数解析为:

id object

:指定 关联者

id value

:指定 被关联者

const void *key

: 被关联者

的 KEY 值,方便后续可以通过该 KEY 值找到该 被关联者

objc_AssociationPolicy policy

: 该参数作用用来表示 被关联者

的引用策略,也就是内存如何进行管理的,可通过上述定义的枚举类型来设置。

id objc_getAssociatedObject(id object, const void *key)

API 可以通过之前设置的 KEY 值,来获取 被关联者

对象,参数解析如下:

id object

: 关联者

对象

const void *key

:要获取的 被关联者

的 KEY 值,一个 关联者

可以被关联多对象,一个 关联者

也可以是 被关联这

,可以通过不同的 KEY 来获取不同的 被关联者

对象。

void objc_removeAssociatedObjects(id object)

该 API 可以移除一个 关联者

对象所有的 被关联者

。当需要移除特定的对象时,我们可以使用 objc_setAssociatedObject

方法并指定 id value

参数对象为空即可。

以上就是关于 Associative(关联)

特性相关的 API 介绍了,下面介绍一下常用的使用场景。

Associative 特性的应用
剪切板的信息复制
在一些时候我们希望用户可以长按文案信息,弹出系统的复制菜单,提供文案信息的复制功能,比如长按 UITableViewCell

提供复制详情的功能,在 iOS 下我们可以使用 UIMenuController

类来显示系统菜单,同时为该 UITableViewCell

添加长按手势,代码如下:

// 添加长按手势
UILongPressGestureRecognizer *longPressGR = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handleLongPress:)];

[cell addGestureRecognizer:longPressGR];

// 手势处理

- (void)handleLongPress:(UILongPressGestureRecognizer *) longPressGR {

    UIMenuController *menu = [UIMenuController sharedMenuController];

    [menu setTargetRect:longPressGR.view.frame inView:self.view];

    [menu setMenuVisible:YES animated:YES];

}

// UIMenuController 相关

- (BOOL)canBecomeFirstResponder {

    return YES;

}

- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {

    if ( action == @selector(copy:) ) {

        return YES;

    }

    return NO;

}

- (void)copy:(UIMenuController *)menu {

    UIPasteboard *pasteboard = [UIPasteboard generalPasteboard];

}

从上述代码可以看到,复制信息的逻辑处理是在 copy:

方法中,但是在该方法中,并不能访问到 cell.detailTextLabel

对象,在该场景中,我们可以使用 Associative

特性将 UITableViewCell

对象关联到 UIMenuController

对象中,再在 copy:

方法中获取到被关联对象,从而获取到 UITableViewCell

对象,进而访问 cell.detailTextLabel.text

。添加代码如下:

// 处理手势时,添加如下代码
objc_setAssociatedObject(menu, @"UITableViewCell", longPressGR.view, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

// 处理 copy 时,添加如下代码,来获取被关联的 UITableViewCell 对象

objc_getAssociatedObject(menu, @"UITableViewCell");

pasteboard.string = cell.detailTextLabel.text;

上述场景中,并不一定非得用 Associative

特性来实现,还有很多可行的方法,这里为大家提供一种方法,并且该方法还算是比较优雅的。

其他一些应用场景
另一个常见的应用场景就是,为一个系统类或是一个第三方的类添加一个属性时,可以结合 Category

为类添加一个属性,当然也可以使用继承来达到目的。在一些特殊场景下,比如想知道一个系统内部对象或者第三方对象是何时被释放时,我们可以为该对象关联一个自定义的对象,并且使用 OBJC_ASSOCIATION_RETAIN_NONATOMIC

来指定内存管理策略,当关联者被释放是,被关联者也会跟着被释放,这样可以在我们自定义的对象中,知道感兴趣的对象何时被释放的。在调试一些内存问题时,该方法还是蛮有用的。

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

推荐阅读更多精彩内容