您的位置:首页 > 移动开发 > Objective-C

对象关联(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects)

2016-04-05 09:04 567 查看
声明:

1、原文地址:http://blog.csdn.net/onlyou930/article/details/9299169

2、本文是我在自身有需求的时候学习了博主的原文,故在原文的基础上做了修改,如有异议请参考原文。

1、需求,即为何要搜索对象关联的使用

在“阳光随访”app项目中的UIViewController (HUD)类别中有三个函数,函数如下:

- (void)showHudInView:(UIView *)view hint:(NSString *)hint dimBackground:(BOOL)dim delegate:(id<MBProgressHUDDelegate>)delegate

{

MBProgressHUD *HUD = [[MBProgressHUD alloc] initWithView:view];

HUD.labelText = hint;

HUD.dimBackground = YES;

HUD.delegate = delegate;

[view addSubview:HUD];

[HUD show:YES];

[self setHUD:HUD];

}

- (MBProgressHUD *)HUD{

return objc_getAssociatedObject(self, HttpRequestHUDKey);

}

- (void)setHUD:(MBProgressHUD *)HUD{

objc_setAssociatedObject(self, HttpRequestHUDKey, HUD, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}
在通过showHudInView函数显示HUD时创建的是局部变量,而且是通过类别创建的HUD,所以也只能是局部变量,但是需求是“要让这个HUD长期存在,在网络请求返回时在请求的代理中可以继续使用HUD”,所以我们必须把HUD关联给使用[b]HUD的对象(即self),使得[b]HUD在self销毁时才销毁。[/b][/b]

关联是指把两个对象相互关联起来,使得其中的一个对象作为源对象的一部分,在源对象和关联对象的引用同时断开的时候关联对象才销毁。

关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的。

2、在类的定义之外为类增加额外的存储空间

使用关联,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。

关联是基于关键字的,因此,我们可以为任何对象增加任意多的关联,每个都使用不同的关键字即可。关联是可以保证被关联的对象在关联对象的整个生命周期都是可用的(在垃圾自动回收环境下也不会导致资源不可回收)。

3、创建关联

创建关联要使用到Objective-C的运行时函数:objc_setAssociatedObject来把一个对象与另外一个对象进行关联。该函数需要四个参数:源对象,关键字,关联的对象和一个关联策略。当然,此处的关键字和关联策略是需要进一步讨论的。

■ 关键字是一个void类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。

■ 关联策略表明了相关的对象是通过赋值,保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。

下面的代码展示了如何把一个字符串关联到一个数组上。

列表7-1 把一个字符串关联到一个数组

[cpp] view
plaincopy

static char overviewKey;

NSArray * array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];

//为了演示的目的,这里使用initWithFormat:来确保字符串可以被销毁

NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];

objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);

[overview release];

//(1) overview仍然是可用的

[array release];

//(2)overview 不可用

在(1)处,字符串overview仍然是可用的,这是因为OBJC_ASSOCIATION_RETAIN策略指明了数组要保有相关的对象。当数组array被销毁的时候,也就是在(2)处overview也就会被释放,因此而被销毁。如果此时还想使用overview,例如想通过log来输出overview的值,则会出现运行时异常。

4、获取相关联的对象

获取相关联的对象时使用Objective-C函数objc_getAssociatedObject。接着上面列表7-1的代码,我们可以使用如下代码来获取与array相关联的字符串:

[cpp] view
plaincopy

NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &oveviewKey);

5、断开关联

断开关联是使用objc_setAssociatedObject函数,传入nil值即可。

接着列表7-1中的程序,我们可以使用如下的代码来断开字符串overview和arry之间的关联:

[cpp] view
plaincopy

objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);

其中,被关联的对象为nil,此时关联策略也就无关紧要了。

使用函数objc_removeAssociatedObjects可以断开所有关联。通常情况下不建议使用这个函数,因为他会断开所有关联。只有在需要把对象恢复到“原始状态”的时候才会使用这个函数。

6、一个完整的实例程序

下面的程序综合了前面的代码.

[cpp] view
plaincopy

#import <Foundation/Foundation.h>

#import <objc/runtime.h>

int main(int argc, const char* argv[])

{

NSAutoreleasePool * pool = [[NSAutoreleasePool] alloc init];

static char overviewKey;

NSArray *array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];

//为了演示的目的,这里使用initWithFormat:来确保字符串可以被销毁

NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];

objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);

[overview release];

NSString *associatedObject = (NSString *)objc_getAssociatedObject(arrray, &overviewKey);

NSLog(@"associatedObject:%@", associatedObject);

objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);

[array release];

[pool drain];

return 0;

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: