最近一直在看SDWebImage的源码,看到UIView分类的时候,想记录一下当时受到的启发:
1.Category:即使在你不知道一个类的源码情况下,同样可以为这个类添加扩展的方法和属性,其中可以很方便地为现有的类添加方法,但却无法直接添加实例变量,要想为一个分类添加属性,就要用到运行时。
2.在Category 类.h中声明的方法,在.m中不实现的话就会报警告:Method definition for 'yourMethodName' not found
3.顺便提一下类扩展 它是一个匿名的分类,类扩展声明必须在.m文件中;可以声明变量、添加属性、方法;在类扩展中声明的方法,不实现的话就会报警告Method definition for 'yourMethodName' not found.
接下来看UIView的这个分类吧。它作用在这里就不再多说,直接看源码吧!
源码如下
精髓就在operationDictionary方法当中
#import "UIView+WebCacheOperation.h"
#import "objc/runtime.h"
static char loadOperationKey;
@implementation UIView (WebCacheOperation)
- (void)sd_setImageLoadOperation:(id)operation forKey:(NSString *)key {
//先取消,在保存
[self sd_cancelImageLoadOperationWithKey:key];
NSMutableDictionary *operationDictionary = [self operationDictionary];
[operationDictionary setObject:operation forKey:key];
}
//key:@"UIImageViewImageLoad"
- (void)sd_cancelImageLoadOperationWithKey:(NSString *)key {
// Cancel in progress downloader from queue
//从队列中取消正在下载的downloader
NSMutableDictionary *operationDictionary = [self operationDictionary];
id operations = [operationDictionary objectForKey:key];
if (operations) {
//执行cancel
if ([operations isKindOfClass:[NSArray class]]) {
for (id <SDWebImageOperation> operation in operations) {
if (operation) {
[operation cancel];
}
}
} else if ([operations conformsToProtocol:@protocol(SDWebImageOperation)]){
[(id<SDWebImageOperation>) operations cancel];
}
//删除
[operationDictionary removeObjectForKey:key];
}
}
- (void)sd_removeImageLoadOperationWithKey:(NSString *)key {
NSMutableDictionary *operationDictionary = [self operationDictionary];
[operationDictionary removeObjectForKey:key];
}
- (NSMutableDictionary *)operationDictionary {
/*
objc_setAssociatedObject作用是对已存在的类在扩展中添加自定义的属性
这个loadOperationKey 的定义是:static char loadOperationKey;
它对应的绑定在UIView扩展中的属性是operations(NSMutableDictionary类型)
*/
NSMutableDictionary *operations = objc_getAssociatedObject(self, &loadOperationKey);
if (operations) {
return operations;
}
operations = [NSMutableDictionary dictionary];
/*
typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy) {
OBJC_ASSOCIATION_ASSIGN = 0,
OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1,
OBJC_ASSOCIATION_COPY_NONATOMIC = 3,
OBJC_ASSOCIATION_RETAIN = 01401,
OBJC_ASSOCIATION_COPY = 01403
};
*/
objc_setAssociatedObject(self, &loadOperationKey, operations, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
return operations;
}
2018年3月15日更新
看了殿神的深入解析 ObjC 中方法的结构之后在这里再添加一下感悟:
1.类在内存中的位置是在编译期间决定的,在之后(这里应该是动态的)修改代码(添加属性、方法),也不会改变内存中的位置。
2.类的方法、属性以及协议在编译期间存放到了“错误”的位置,直到 realizeClass执行之后,才放到了class_rw_t指向的只读区域class_ro_t,这样我们即可以在运行时为class_rw_t添加方法,也不会影响类的只读结构。
在 class_ro_t 中的属性在运行期间就不能改变了,再添加方法时,会修改class_rw_t中的methods列表,而不是class_ro_t中的 baseMethods。