一些学习runtime时的疑惑
1. self和super
- 先看官方文档
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(),用于调用基类对象/类方法
- 使用区别
- 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