self,super,superClass引出的关于ivarList内存在运行时底层的一个问题

首先先看示例代码:

- (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中对于一个实例变量可能会因为有多个(子类或者当前类)对象的产生而生成多个地址保存不同对象中的同一属性值。(这句话不知对不对,还请大神指教,谢谢)。

你可能感兴趣的:(self,super,superClass引出的关于ivarList内存在运行时底层的一个问题)