Runtime常用接口方法解析

作者: 奋拓达 | 来源:发表于2018-05-14 16:53 被阅读4次
    RunTime简称运行时,就是系统在运行的时候的一些机制,其中最主要的是消息机制
    OC与C语言在函数调用上的对比
    
    C语言

    函数的调用在编译的时候就决定调用哪个函数,编译完成之后直接顺序执行,无任何二义性。
    C语言在编译阶段调用未实现的函数就会报错

    OC语言

    函数的调用成为消息发送。属于动态调用过程。
    在编译的时候并不能决定真正调用哪个函数(事实证明,在编译阶段,OC可以调用任何函数,即使这个函数并未实现,只要声明过就不会报错)

    三种层面上与 Runtime 系统进行交互
    1. 通过 Objective-C 源代码

    多数情况我们只需要编写 OC 代码即可,Runtime 系统自动在幕后搞定一切,还记得简介中如果我们调用方法,编译器会将 OC 代码转换成运行时代码,在运行时确定数据结构和函数

    1. 通过 Foundation 框架的 NSObject 类定义的方法

    Cocoa 程序中绝大部分类都是 NSObject 类的子类,所以都继承了 NSObject 的行为。(NSProxy 类是个例外,它是个抽象超类)

    一些情况下,NSObject 类仅仅定义了完成某件事情的模板,并没有提供所需要的代码。例如 -description 方法,该方法返回类内容的字符串,该方法主要用来调试程序。NSObject 类并不知道子类的内容,所以它只是返回类的名字和对象的地址,NSObject 的子类可以重新实现。

    1. NSObject 还有一些方法可以从 Runtime 系统中获取信息,允许对象进行自我检查如下:
    • class方法返回对象的类;
    • isKindOfClass:isMemberOfClass: 方法检查对象是否存在于指定的类的继承体系中(是否是其子类或者父类或者当前类的成员变量);
    • respondsToSelector: 检查对象能否响应指定的消息;
    • conformsToProtocol:检查对象是否实现了指定协议类的方法;
    • methodForSelector: 返回指定方法实现的地址。

    通过对 runtime 库函数的直接调用

    Runtime 系统是具有公共接口的动态共享库。头文件存放于/usr/include/objc目录下,这意味着我们使用时只需要引入

    获取类:
    Class PersonClass = object_getClass([Person class]);
    
    SEL是selector在 Objc 中的表示:
    SEL oriSEL = @selector(test1);
    
    获取类方法:
    Method oriMethod = Method class_getClassMethod(Class cls , SEL name);
    
    获取实例方法:
    Method class_getInstanceMethod(Class cls , SEL name)
    
    添加方法:
    BOOL addSucc = class_addMethod(xiaomingClass, oriSEL, method_getImplementation(cusMethod), method_getTypeEncoding(cusMethod));
    
    替换原方法实现:
    class_replaceMethod(toolClass, cusSEL, method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));
    
    交换两个方法:
    method_exchangeImplementations(oriMethod, cusMethod);
    
    获取一个类的属性列表(返回值是一个数组):
    objc_property_t *propertyList = class_copyPropertyList([self class], &count);
    
    获取一个类的方法列表(返回值是一个数组):
    Method *methodList = class_copyMethodList([self class], &count);
    
    获取一个类的成员变量列表(返回值是一个数组):
    Ivar *ivarList = class_copyIvarList([self class], &count);
    
    获取成员变量的名字:
    const char *ivar_getName(Ivar v)
    
    获取成员变量的类型:
    const char *ivar_getTypeEndcoding(Ivar v)
    
    获取一个类的协议列表(返回值是一个数组):
    __unsafe_unretained Protocol **protocolList = class_copyProtocolList([self class], &count);
    
    set方法:
    //将值value 跟对象object 关联起来(将值value 存储到对象object 中)
    //参数 object:给哪个对象设置属性
    //参数 key:一个属性对应一个Key,将来可以通过key取出这个存储的值,key 可以是任何类型:double、int 等,建议用char 可以节省字节
    //参数 value:给属性设置的值
    //参数policy:存储策略 (assign 、copy 、 retain就是strong)
    void objc_setAssociatedObject(id object , const void *key ,id value ,objc_AssociationPolicy policy)
    
    利用参数key 将对象object中存储的对应值取出来:
    id objc_getAssociatedObject(id object , const void *key)
    

    小结:以上的所罗列出的接口都是我们在运用runtime时常使用的接口

    相关文章

      网友评论

        本文标题:Runtime常用接口方法解析

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