首先先看示例代码:
- (void)viewDidLoad {
[super viewDidLoad];
Student *stu1 = [[Student alloc]init];
stu1.name = @"stu1";
Student *stu2 = [[Student alloc]init];
stu2.name = @"stu2";
[stu1 test];
[stu2 test];
}
// test是student实例方法:说明一下:student继承自person,name是person的实例
- (void)test{
NSLog(@"%@ : %p",self.name,self.name);
NSLog(@"%@ super : %p",self.name ,[super name]);
NSLog(@"super : %p",[self superclass]);
}
输出:
2016-05-03 00:15:27.394 runtimeTest[2772:351176] stu1 : 0x10e687220
2016-05-03 00:15:27.395 runtimeTest[2772:351176] stu1 super : 0x10e687220
2016-05-03 00:15:27.395 runtimeTest[2772:351176] super : 0x10e6886a0
2016-05-03 00:15:27.395 runtimeTest[2772:351176] stu2 : 0x10e687240
2016-05-03 00:15:27.395 runtimeTest[2772:351176] stu2 super : 0x10e687240
2016-05-03 00:15:27.395 runtimeTest[2772:351176] super : 0x10e6886a0
首先,不管是在执行self.name [super name] 在oc中都是执行消息机制。那么self.name会被翻译成
objc_msgSend(id self,SEL _cmd, ...)
也就是当前对象作为接受者,并且先在当前类(也就是student类)的methodList中寻找函数。
然而执行[super name]的时候会被翻译成
id objc_msgSendSuper(struct objc_super *super, SEL op, ...)
其中objc_super结构如下:
struct objc_super {
id receiver;
Class superClass;
};
可以看出消息接受者为receiver还是self,但是会去当前类的superClass指针所指向的类(即父类)的methodList中去寻找。
那么,总结:
1. stu1和stu super 的name的地址相同可以说明继承关系并不会给子类增加实例变量,只会增加superClass指针,让在当前类找不到实例变量的时候再去父类ivarList查找。
2. stu1 和 stu2 或 stu1 super 和stu2 super 地址不同则说明返回的name所在地址不同(这句是废话,地址相同值不也一样了吗)。但是,superClass指针指向的类对象地址相同,那么就说明在类对象中的ivarList中对于一个实例变量可能会因为有多个(子类或者当前类)对象的产生而生成多个地址保存不同对象中的同一属性值。(这句话不知对不对,还请大神指教,谢谢)。
网友评论