内存管理iOS基本功将来跳槽用

category 和 extension 有什么区别?categ

2019-01-28  本文已影响85人  iOS猿_员

原文:iOS面试题大全

extension 在编译期决定,它就是类的一部分,在编译期和头文件里的 @interface 以及实现文件里的 @implement 一起形成一个完整的类,它伴随类的产生而产生,亦随之一起消亡。extension 一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加 extension,所以你无法为系统的类比如 NSString 添加 extension。

但是 category 则完全不一样,它是在运行期决定的。就 category 和 extension 的区别来看,我们可以推导出一个明显的事实,extension 可以添加实例变量,而 category 是无法添加实例变量的(因为在运行期,对象的内存布局已经确定,如果添加实例变量就会破坏类的内部布局,这对编译型语言来说是灾难性的)。

category 的加载是发生在运行时,加载 category 的过程:

其中需要注意的是:

在类和 category 中都可以有 +load方法,那么有两个问题:

上面讲到的方法覆盖,还有一个补充问题:怎么调用到原来类中被 category 覆盖掉的方法?对于这个问题,我们已经知道 category 其实并不是完全替换掉原来类的同名方法,只是 category 在方法列表的前面而已,所以我们只要顺着方法列表找到最后一个对应名字的方法,就可以调用原来类的方法:

    // 假设被覆盖的方法名叫 printName。
    Class currentClass = [MyClass class];
    MyClass *my = [[MyClass alloc] init];

    if (currentClass) {
        unsigned int methodCount;
        Method *methodList = class_copyMethodList(currentClass, &methodCount);
        IMP lastImp = NULL;
        SEL lastSel = NULL;
        for (NSInteger i = 0; i < methodCount; i++) {
            Method method = methodList[i];
            NSString *methodName = [NSString stringWithCString:sel_getName(method_getName(method)) encoding:NSUTF8StringEncoding];
            if ([@"printName" isEqualToString:methodName]) {
                lastImp = method_getImplementation(method);
                lastSel = method_getName(method);
            }
        }
        typedef void (*fn)(id,SEL);

        if (lastImp != NULL) {
            fn f = (fn) lastImp;
            f(my, lastSel);
        }
        free(methodList);
    }
上一篇 下一篇

猜你喜欢

热点阅读