我们刚学习OC的时候知道类目是不可以添加属性的,其实这不是绝对的,我们可以通过objc_setAssociatedObject动态添加属性,和类进行关联,那么首先了解下什么是关联:
关联是指把两个对象相互关联起来,使得其中的一个对象作为另外一个对象的一部分。
关联特性只有在Mac OS X V10.6以及以后的版本上才是可用的。
在类的定义之外为类增加额外的存储空间
使用关联,我们可以不用修改类的定义而为其对象增加存储空间。这在我们无法访问到类的源码的时候或者是考虑到二进制兼容性的时候是非常有用。关联是基于关键字的,因此,我们可以为任何对象增加任意多的关联,每个都使用不同的关键字即可。关联是可以保证被关联的对象在关联对象的整个生命周期都是可用的(在垃圾自动回收环境下也不会导致资源不可回收)。
创建关联
创建关联要使用到Objective-C的运行时函数:objc_setAssociatedObject来把一个对象与另外一个对象进行关联。该函数需要四个参数:源对象,关键字,关联的对象和一个关联策略。当然,此处的关键字和关联策略是需要进一步讨论的。■ 关键字是一个void类型的指针。每一个关联的关键字必须是唯一的。通常都是会采用静态变量来作为关键字。
■ 关联策略表明了相关的对象是通过赋值,保留引用还是复制的方式进行关联的;还有这种关联是原子的还是非原子的。这里的关联策略和声明属性时的很类似。这种关联策略是通过使用预先定义好的常量来表示的。
下面的代码展示了如何把一个字符串关联到一个数组上。
列表7-1 把一个字符串关联到一个数组
- 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 不可用
获取相关联的对象
获取相关联的对象时使用Objective-C函数objc_getAssociatedObject。接着上面列表7-1的代码,我们可以使用如下代码来获取与array相关联的字符串:- NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &oveviewKey);
断开关联
断开关联是使用objc_setAssociatedObject函数,传入nil值即可。接着列表7-1中的程序,我们可以使用如下的代码来断开字符串overview和arry之间的关联:
- objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
使用函数objc_removeAssociatedObjects可以断开所有关联。通常情况下不建议使用这个函数,因为他会断开所有关联。只有在需要把对象恢复到“原始状态”的时候才会使用这个函数。
一个完整的实例程序
下面的程序综合了前面的代码.- #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;
- }
1. 给NSObject类动态添加属性
h定义部分
- @interface UIWebView (LoadProgress)
- @property (nonatomic, assign) NSInteger resourceCount;
- @end
m实现部分
首先要定义一个全局的key
- // resourceCount object keys
- static void *s_resourceCountKey = &s_resourceCountKey;
- //static void *s_resourceCountKey = "s_resourceCountKey";
初始
- @implementation UIWebView (LoadProgress)
- @dynamic resourceCount;
实现
- #pragma mark Accessors and mutators
- - (NSInteger)resourceCount
- {
- NSNumber *resourceCountNumber = objc_getAssociatedObject(self, s_resourceCountKey);
- if (! resourceCountNumber)
- {
- return 0;
- }
- else
- {
- return [resourceCountNumber integerValue];
- }
- }
- - (void)setResourceCount:(NSInteger)rCount
- {
- objc_setAssociatedObject(self, s_resourceCountKey, [NSNumber numberWithInteger:rCount], OBJC_ASSOCIATION_RETAIN_NONATOMIC);
- }
这样就可以直接使用了
webView.resourceCount = 10;
3. 替换或变更NSObject类方法Method
基本替换方法
- // 替换类方法
- // frome: CoconutKit
- IMP HLSSwizzleClassSelector(Class clazz, SEL selector, IMP newImplementation)
- {
- // Get the original implementation we are replacing
- Class metaClass = objc_getMetaClass(class_getName(clazz));
- Method method = class_getClassMethod(metaClass, selector);
- IMP origImp = method_getImplementation(method);
- if (! origImp) {
- return NULL;
- }
- class_replaceMethod(metaClass, selector, newImplementation, method_getTypeEncoding(method));
- return origImp;
- }
- // 替换实例方法
- IMP HLSSwizzleSelector(Class clazz, SEL selector, IMP newImplementation)
- {
- // Get the original implementation we are replacing
- Method method = class_getInstanceMethod(clazz, selector);
- IMP origImp = method_getImplementation(method);
- if (! origImp) {
- return NULL;
- }
- class_replaceMethod(clazz, selector, newImplementation, method_getTypeEncoding(method));
- return origImp;
- }
新方法定义
- // Original implementation of the methods we swizzle
- static id (*s_UIWebView__identifierForInitialRequest_Imp)(id, SEL, id, id, id) = NULL;
- // Swizzled method implementations
- static id swizzled_UIWebView__identifierForInitialRequest_Imp(UIWebView *self, SEL _cmd, id webView, id initialRequest, id dataSource);
方法初始化需要写在类方法+ (void)load中
- + (void)load
- {
- s_UIWebView__identifierForInitialRequest_Imp = (id (*)(id, SEL, id, id, id))HLSSwizzleSelector(self, @selector(webView:identifierForInitialRequest:fromDataSource:), (IMP)swizzled_UIWebView__identifierForInitialRequest_Imp);
- }
实现部分
- #pragma mark Swizzled method implementations
- static id swizzled_UIWebView__identifierForInitialRequest_Imp(UIWebView *self, SEL _cmd, id webView, id initialRequest, id dataSource)
- {
- // 调用原方法
- (*s_UIWebView__identifierForInitialRequest_Imp)(self, _cmd, webView, initialRequest, dataSource);
- [self setResourceCount:self.resourceCount+1];
- return [NSNumber numberWithInteger:self.resourceCount];
- }
- // [self implementsProtocol:@protocol(UIActionSheetDelegate)]
- // frome: CoconutKit
- - (BOOL)implementsProtocol:(Protocol *)protocol
- {
- // Only interested in optional methods. Required methods are checked at compilation time
- unsigned int numberOfMethods = 0;
- struct objc_method_description *methodDescriptions = protocol_copyMethodDescriptionList(protocol, NO /* optional only */, YES, &numberOfMethods);
- for (unsigned int i = 0; i < numberOfMethods; ++i) {
- struct objc_method_description methodDescription = methodDescriptions[i];
- SEL selector = methodDescription.name;
- if (! class_getInstanceMethod([self class], selector)) {
- NSString *selectorString = [NSString stringWithCString:sel_getName(selector) encoding:NSUTF8StringEncoding];
- NSString *protocolName = [NSString stringWithCString:protocol_getName(protocol) encoding:NSUTF8StringEncoding];
- HLSLoggerInfo(@"Class %@ does not implement method %@ of protocol %@", [self className], selectorString, protocolName);
- selectorString = nil; // Just to remove unused variable warnings
- protocolName = nil;
- return NO;
- }
- }
- return YES;
- }