美文网首页iOS
你要理解的Runtime知识点

你要理解的Runtime知识点

作者: FITZ9311 | 来源:发表于2015-11-15 15:55 被阅读252次

    博客地址:张飞的技术博客
    在学习Runtime的过程中,有不少的知识点是平时难以接触到的。所以刚开始接触Runtime的时候会碰到好些不熟悉的小知识点。下面就随我扒一扒Runtime中那些需要知道的小知识点吧!

    SEL

    SEL是一种类型,该类型表示对方法的一种封装,平时我们说的选择器selector就是它啦!Objective-C在编译时候呢,会依据每一个方法的名字、参数序列,生成一个唯一的整型标识(Int类型的地址),这个标识就是SEL。也就是说每一个SEL对应一个方法,根据一个SEL就能找到一个方法,然后对它进行调用等操作。其实我们也可以理解为SEL是一个方法的指针(实际上不是)。来看看它的真面目吧,在objc.h的49行处可以发现:

    /// An opaque type that represents a method selector.
    typedef struct objc_selector *SEL;
    

    这样看来SEL的本质是一个objc_selector类型的结构体指针。那该如何生成一个SEL类型的数据呢?编译器提供的有选择器@selector,NSSelectorFromString()方法,sel_registerName函数。

    SEL methodSel1 = @selector(method);
    SEL methodSel2 = NSSelectorFromString(method);
    ```
    >NOTE:只要方法名一样,那么这个方法的SEL就一样,不管同名的方法是有没有关系(继承),即使在不同的类方法名一样那么这个方法的SEL也是一样的。但是同一个类中不能存在同名方法(即使参数类型一样也不行,当然参数个数不一样可以),在不同的类里面是可以有同名方法的,不同类的实例对象执行相同的selector时,会在各自的方法列表中去根据selector去寻找自己对应的IMP。
    
    本质上,SEL只是一个指向方法的指针(准确的说,只是一个根据方法名hash化了的KEY值,能唯一代表一个方法),它的存在只是为了加快方法的查询速度。
    
    ###IMP
    上面讲了,根据SEL能寻找到函数,其实IMP才是真正指向函数的指针,它指向了函数实现的首地址。其实SEL就是为了查找方法的IMP的。取得IMP后,我们就获得了执行这个方法代码的入口点,此时,我们就可以像调用普通的C语言函数一样来使用这个函数指针了。
    
    >NOTE:通过取得IMP,我们可以跳过Runtime的消息传递机制,直接执行IMP指向的函数实现,这样省去了Runtime消息传递过程中所做的一系列查找操作,会比直接向对象发送消息高效一些。
    
    ###Method
    Method是一个结构体,用于表示类中定义的方法。实际上相当于在SEL和IMP之间作了一个映射。有了SEL,我们便可以找到对应的IMP,从而调用方法的实现代码。
    >方法操作相关函数:
    
    ``` Objective-C
    // 调用指定方法的实现
    id method_invoke ( id receiver, Method m, ... );
    // 调用返回一个数据结构的方法的实现
    void method_invoke_stret ( id receiver, Method m, ... );
    // 获取方法名
    SEL method_getName ( Method m );
    // 返回方法的实现
    IMP method_getImplementation ( Method m );
    // 获取描述方法参数和返回值类型的字符串
    const char * method_getTypeEncoding ( Method m );
    // 获取方法的返回值类型的字符串
    char * method_copyReturnType ( Method m );
    // 获取方法的指定位置参数的类型字符串
    char * method_copyArgumentType ( Method m, unsigned int index );
    // 通过引用返回方法的返回值类型字符串
    void method_getReturnType ( Method m, char *dst, size_t dst_len );
    // 返回方法的参数的个数
    unsigned int method_getNumberOfArguments ( Method m );
    // 通过引用返回方法指定位置参数的类型字符串
    void method_getArgumentType ( Method m, unsigned int index, char *dst, size_t dst_len );
    // 返回指定方法的方法描述结构体
    struct objc_method_description * method_getDescription ( Method m );
    // 设置方法的实现
    IMP method_setImplementation ( Method m, IMP imp );
    // 交换两个方法的实现
    void method_exchangeImplementations ( Method m1, Method m2 );
    ```
    上面这些函数在通过Runtime实现一些东西的时候或者在看别人的代码的时候或许能用到,只需要知道有这么一些函数就可以了。
    
    ###Class
    Class在我的一篇文章中有详细的讲解,请移步[OC中的类是怎么来的?](),这里不在赘诉了。
    
    ###_cmd
    通过_cmd这个关键字能获取当前方法的`SEL`。看个栗子吧:
    ```
    - (void)testCmd {
        SEL currentMethodSel = _cmd;
        NSLog(@"currentSel is :%s",(char *)currentMethodSel); 
    }
    ```
    上面的打印结果为:
    ```
    currentSel is :testCmd
    ```
    
    ###objc_msgSend
    在OC中,对方法的调用被称之为发送消息。其实我们对方法的调用最后都会被编译为下面的这种形式:
    >对于函数:
    
    ```
    [target method1];
    [target method2:var];
    ```
    
    >没有参数:
    
    ```
    objc_msgSend(target,@selector(method1));
    ```
    
    >有参数:
    
    ```
    objc_msgSend(target,@selector(method2:),var);
    ```
    方法的调用过程在我的上一篇文章[OC中的类是怎么来的?]()
    
    ###结尾
    有人说Runtime是OC的本质和精华,虽然我们在项目中不经常用,但是理解还是有很大的用处的。喜欢我的文章就给我鼓励吧!
    ![](http:https://img.haomeiwen.com/i465386/2f3e022157448deb.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/200)

    相关文章

      网友评论

      本文标题:你要理解的Runtime知识点

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