標籤:
Associated Objects(關聯對象)或者叫作關聯引用(Associative References),是作為Objective-C 2.0 運行時功能被引入到 Mac OS X 10.6 Snow Leopard(及iOS4)系統。與它相關在<objc/runtime.h>中有3個C函數,它們可以讓對象在運行時關聯任何值:
OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)OBJC_EXPORT void objc_removeAssociatedObjects(id object)
為什麼這幾個方法很有用呢?因為開發人員可以通過它們在分類中給已存在的類中添加自訂屬性。
key 值
關於前兩個函數中的 key 值是我們需要重點關注的一個點,這個 key 值必須保證是一個對象層級(為什麼是對象層級?看完下面的章節你就會明白了)的唯一常量。一般來說,有以下三種推薦的 key 值:
- 聲明
static char kAssociatedObjectKey; ,使用 &kAssociatedObjectKey 作為 key 值;
- 聲明
static void *kAssociatedObjectKey = &kAssociatedObjectKey; ,使用 kAssociatedObjectKey 作為 key 值;
- 用
selector ,使用 getter 方法的名稱作為 key 值。
我個人最喜歡的(沒有之一)是第 3 種方式,因為它省掉了一個變數名,非常優雅地解決了計算科學中的兩大世界難題之一(命名)。
關聯策略
在給一個對象添加關聯對象時有五種關聯策略可供選擇:
/** * OBJC_ASSOCIATION_ASSIGN 等價於@property (assign) , @property (unsafe_unretained) 弱引用關聯對象 * OBJC_ASSOCIATION_RETAIN_NONATOMIC 等價於@property (strong, nonatomic) 強引用關聯對象,且為非原子操作 * OBJC_ASSOCIATION_COPY_NONATOMIC 等價於@property (copy, nonatomic) 複製關聯對象,且為非原子操作 * OBJC_ASSOCIATION_RETAIN 等價於@property (strong, atomic) 強引用關聯對象,且為原子操作 * OBJC_ASSOCIATION_COPY 等價於@property (copy, atomic) 複製關聯對象,且為原子操作 其中,第 2 種與第 4 種、第 3 種與第 5 種關聯策略的唯一差別就在於操作是否具有原子性。 */ objc_setAssociatedObject(button, buttonKey, _imgView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
例子:
1、倒入標頭檔,聲明key:
#import <objc/runtime.h>char* const buttonKey = "buttonKey";
2、設定關聯:
if (groupModel.isOpened) { UIImageView * _imgView = [[UIImageView alloc]initWithFrame:CGRectMake(10, (44-16)/2, 14, 16)]; [_imgView setImage:[UIImage imageNamed:@"ico_list"]]; [sectionView addSubview:_imgView]; CGAffineTransform currentTransform = _imgView.transform; CGAffineTransform newTransform = CGAffineTransformRotate(currentTransform, M_PI/2); // 在現在的基礎上旋轉指定角度 _imgView.transform = newTransform; objc_setAssociatedObject(button, buttonKey, _imgView, OBJC_ASSOCIATION_RETAIN_NONATOMIC); }else{ UIImageView * _imgView = [[UIImageView alloc]initWithFrame:CGRectMake(10, (44-16)/2, 14, 16)]; [_imgView setImage:[UIImage imageNamed:@"ico_list"]]; [sectionView addSubview:_imgView]; objc_setAssociatedObject(button, buttonKey, _imgView, OBJC_ASSOCIATION_RETAIN_NONATOMIC); }
3、擷取關聯:
- (void)buttonPress:(UIButton *)sender//headButton點擊{ GroupModel *groupModel = dataSource[sender.tag]; UIImageView *imageView = objc_getAssociatedObject(sender,buttonKey); if (groupModel.isOpened) { [UIView animateWithDuration:0.3 delay:0.0 options:UIViewAnimationOptionTransitionNone animations:^{ CGAffineTransform currentTransform = imageView.transform; CGAffineTransform newTransform = CGAffineTransformRotate(currentTransform, -M_PI/2); // 在現在的基礎上旋轉指定角度 imageView.transform = newTransform; } completion:^(BOOL finished) { }]; }else{ [UIView animateWithDuration:0.3 delay:0.0 options: UIViewAnimationOptionAllowUserInteraction |UIViewAnimationOptionCurveLinear animations:^{ CGAffineTransform currentTransform = imageView.transform; CGAffineTransform newTransform = CGAffineTransformRotate(currentTransform, M_PI/2); // 在現在的基礎上旋轉指定角度 imageView.transform = newTransform; } completion:^(BOOL finished) { }]; } groupModel.isOpened = !groupModel.isOpened; [expandTable reloadSections:[NSIndexSet indexSetWithIndex:sender.tag] withRowAnimation:UITableViewRowAnimationAutomatic];}
iOS關聯對象