美文网首页
runtime(三)-一些补充

runtime(三)-一些补充

作者: George_Luofz | 来源:发表于2018-04-06 06:59 被阅读12次

    一些学习runtime时的疑惑

    1. self和super
    1. 先看官方文档

    self:
    Whenever you’re writing a method implementation, you have access to an important hidden value, self. Conceptually, self is a way to refer to “the object that’s received this message.” It’s apointer, just like the greeting value above, and can be used to call a method on the current receiving object.
    super:
    There’s anotherimportant keyword available to you in Objective-C, called super. Sending a message to super is a way to call through to a method implementation defined by a superclass further up the inheritance chain. The most common use of super is when overriding a method.

    由官方文档可知
    self 是一个隐藏参数,用于调用当前对象/类方法
    super 是一个编译器参数,会被替换成objc_msgSendSuper(),用于调用基类对象/类方法

    1. 使用区别
    • self 可以单独打印,super必须结合方法使用,比如下边代码会报错,undeclared identifier.
    - (void)_test_self_super{
        NSLog(@"self:%@ super:%@ %s",self,super,__func__);
    }
    
    • [self class]和[super class]打印结果一样(在都没有重写的情况下一样),需要结合编译器替换成的objc_msgSendSuper方法和objc_super结构体定义理解:
    
    id objc_msgSend(id self, SEL op, ...)
    
    id objc_msgSendSuper(struct objc_super *super, SEL op, ...)
    
    struct objc_super {
    id receiver;
    Class superClass;
    };
    

    由上边可以明白objc_msgSendSuper传入的super->receiver就是self和objc_msgSend传入的参数一样,而SEL参数也就是@selector(class)在当前类和父类都没有实现的情况下也都会调用根元类NSObject的class方法,所以就一样了;然后看看NSObject的class实现源码就更清晰了,源码如下

    - (Class)class {
        return object_getClass(self);
    }
    Class object_getClass(id obj)
    {
        if (obj) return obj->getIsa(); 
        else return Nil;
    }
    

    可以看到object_getClass返回的就是obj的isa,也就是类对象,所以一个Test类(继承与NSObject)中打印[self class]和[super class]返回的的都是是Test 类对象

    但是如果我们重写了当前的class方法,就不一样了,比如:

    - (Class)class{ //重写了TestRuntimeForward类的class方法
        return [NSObject class];
    }
    - (void)test_self_super{
        NSLog(@"\nself class:%@\nsuper class:%@",[self class],[super class]);
    }
    // 打印输出,可以看到
    2018-04-06 06:54:49.826058+0800 iOSLearnigDemo[8110:789323] 
    self class:NSObject
    super class:TestRuntimeForward
    
    • self = [super init]干嘛?
      在当前类初始化前,先做基类初始化,所有类能够使用,都是需要初始化的;我们能在当前对象中使用class方法、performSelector方法等等,其实就是通过NSObject调用的,你不做基类的初始化,还能调用这些方法吗
      直接[super init]不赋值给self也可以
      参考:OC中的self和super
    2. 理解NSMethodSignature

    由命名可以看出这是一个方法签名,封装了方法的参数和返回值,可以用它类初始化NSInvocation

    + (NSInvocation *)invocationWithMethodSignature:(NSMethodSignature *)sig;
    
    3. 理解NSInvocation

    封装了target、方法参数、方法返回值等等信息,相当于将一个方法的执行信息封装成一个对象,类似于cmd模式
    很多利用消息转发机制的骚操作,都是使用NSInvocation实现的,例如著名的JSPatch,因为它的信息最全,功能最强大
    上段代码:

    - (void)_test_invocation{
        NSMethodSignature *signature = [NSMethodSignature signatureWithObjCTypes:"v@::"]; //只能用这种方式生成一个methodSignature,不能使用[self methodSignatureForSelector:]方法,因为self中该方法没实现,会调用父类的该方法,返回nil;第一个:表示selector,第二个:表示selector中的第一个参数,总共有3个参数(两个默认,一个自定义)
        NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
        [invocation setTarget:self];
        [invocation setSelector:@selector(_test_invocation_print:)];
        int a = 2;
        [invocation setArgument:&a atIndex:2]; //从2开始
    //    [invocation retainArguments]; 
        [invocation invoke];
    }
    - (void)_test_invocation_print:(int)a{
        NSLog(@"%s a=%d",__func__,a);
    }
    输出结果:
    2018-04-06 07:22:00.734305+0800 iOSLearnigDemo[8408:808413] -[RuntimeViewController _test_invocation_print:] a=2
    

    相关文章

      网友评论

          本文标题:runtime(三)-一些补充

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