美文网首页iOS面试总结
OC分类二(Category)

OC分类二(Category)

作者: iLeooooo | 来源:发表于2019-02-15 18:06 被阅读14次

+load


M、题:Category中有load方法吗?load方法什么时候调用?load方法能继承吗?

答:有;+load方法会在Runtime加载类,分类的时候调用;+load方法可以继承,但是一般情况下不会主动去调用+load方法,都是让系统自动调用;每个类,分类的+load方法,在程序运行过程中只调用一次。+load方法是Runtime在加载类,分类的时候直接取出类的+load方法直接调用的,+test等一般的方法调用是通过objc_msgSend发送消息调用的(消息发送机制);两种调用有本质区别,消息发送机制会通过superclass指针一步步找方法实现,一旦找到即调用。类和分类的+load方法每个都会调用是因为是直接通过Runtime拿到每个类和分类的函数指针,指向+load函数,拿到+load函数地址,找到+load函数的实现,直接调用。


objc4源码解读过程 : objc-os.mm

_objc_init
load_images
prepare_load_methods
schedule_class_load
add_class_to_loadable_list
add_category_to_loadable_list

call_load_methods
call_class_loads
call_category_loads
(*load_method)(cls, SEL_load)
struct loadable_class {
    Class cls;  // may be nil
    IMP method;    // 指向+load方法
};

struct loadable_category {
    Category cat;  // may be nil
    IMP method;    // 指向分类的+load方法
};
调用顺序:
  1. 先调用类的+load
    • 按照编译先后顺序调用,先编译,先调用
    • 调用子类的+load方法之前,会先调用父类的+load
  2. 再调用分类的+load
    • 按照编译先后顺序调用

+initialize


N、题:load、initialize方法的区别是什么?它们在Category中的调用顺序?以及出现继承是它们的调用顺序?

答:+initialize方法会在类第一次接收到消息是调用。区别:+initialize是通过objc_msgSend进行调用的,所以有以下特点:1.如果子类没有实现+initialize,会调用父类的+initialize(所以父类的+initialize可能会被调用多次)2.如果分类实现了+initialize,就覆盖类本身的+initialize调用。+load方法不是通过消息发送机制调用的(调用方式见上)。调用顺序是按照Category的原来的顺序调用的,即后编译,先调用。出现继承,如果自己实现了+initialize调用自己的,如果自己没有实现+initialize,调用父类的+initialize(调用父类的+initialize的顺序按照后编译先调用顺序执行)。


load、initialize的区别:

  1. 调用方法不同:
    • load是根据函数地址直接调用
    • initialize是通过objc_msgSend(消息发送机制)调用
  2. 调用时刻:
    • load方法是在Runtime加载类,分类的时候调用(只会调用1次)
    • initialize是类第一次接收到消息是调用,每一个类只会initialize一次(父类的+initialize可能会被调用多次)

load、initialize的调用顺序:

  1. load:
    1> 先调用类的load

    • 先编译的类,优先调用load
    • 调用子类的load之前,会先调用父类的load

    2> 再调用分类的load

    • 先编译的类,优先调用load
  2. initialize
    1> 先初始化父类
    2> 再初始化子类(可能最终调用的是父类的initialize[子类initialize未实现])

objc4源码解读过程objc-msg-arm64.s

objc_msgSend

objc-runtime-new.mm

class_getInstanceMethod
lookUpImpOrNil
lookUpImpOrForward
_class_initialize
callInitialize
objc_msgSend(cls, SEL_initialize) 
调用顺序:
  1. 先调用父类的+initialize,再调用子类的+initialize
    (先初始化父类,再初始化子类,每个类只会初始化一次)

objc_msgSend:

  • isa -> 类对象/元类对象,寻找方法,调用方法
  • superclass -> 类对象/元类对象,寻找方法,调用方法
  • superclass -> 类对象/元类对象,寻找方法,调用方法
    ...

相关文章

网友评论

    本文标题:OC分类二(Category)

    本文链接:https://www.haomeiwen.com/subject/qqjpeqtx.html