热门搜索 :
考研考公
您的当前位置:首页正文

iOS关于分类和延展

来源:东饰资讯网

最近一直在看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日更新

类结构.png

1.类在内存中的位置是在编译期间决定的,在之后(这里应该是动态的)修改代码(添加属性、方法),也不会改变内存中的位置。
2.类的方法、属性以及协议在编译期间存放到了“错误”的位置,直到 realizeClass执行之后,才放到了class_rw_t指向的只读区域class_ro_t,这样我们即可以在运行时为class_rw_t添加方法,也不会影响类的只读结构。
在 class_ro_t 中的属性在运行期间就不能改变了,再添加方法时,会修改class_rw_t中的methods列表,而不是class_ro_t中的 baseMethods。

Top