码迷,mamicode.com
首页 > 其他好文 > 详细

黑马程序员 OC self指针

时间:2015-04-13 23:03:36      阅读:122      评论:0      收藏:0      [点我收藏+]

标签:objective c   黑马程序员   

当调用对象方法时,编译器都会默认传入一个指向本对象的指针。所以不同的对象都会调用到正确的成员变量。这个指针就是self,它的值就是new时在堆中分配内存的首地址。那么这个self在方法调用时是存储在栈中么?上代码调试来看一下

#import <Foundation/Foundation.h>

@interface Person : NSObject
{
    int _age;
}

- (void)setAge:(int)age;
- (int)age;

- (void)test:(int)age;
@end



int main(int argc, const char * argv[])
{
    
    Person *p = [Person new];
    
    NSLog(@" p = %p", p);
    
    [p setAge:10];
    
    [p test:20];
    
    return 0;
}



@implementation Person

- (void)setAge:(int)age
{
    _age = age;
}

- (int)age
{
    return _age;
}

- (void)test:(int)age
{
    int _age = 20;
    NSLog(@"局部变量_age = %d", _age);
    NSLog(@"年龄:%d", self->_age);
}

@end

通过输出p中的地址值和调试窗口观察 self中的值和p中的值是一样。也就是self指向的对象在堆中内存空间的首地址

技术分享

在运行到_age=age;语句时断下。观察反汇编代码和相关寄存器值 我们发现
寄存器rsi中的值和self中的值 是一样的。 edx = 10(也就是rdx寄存器的低32位);  rsi = 8;

movl   %edx, (%rsi,%rdi) //这条汇编语句的意思就是把 edx中的值复制到 rsi + rdi 的地址所指向的内存中去。并且操作数据长度为4字节。
技术分享

单步下向执行一条汇编语句。断在popq   %rbp处。此时movl   %edx, (%rsi,%rdi) 已经执行完毕。而断点所指向的语句是将要执行但还未执行的语句
打开self所指向的内存窗口再次观察

技术分享

我们发现self + 8地址处4字节空间已经被赋值为 00 00 00 0A;所以_age最终被设置为10;也就是说成员变量的地址是在self+8处开始的。

那这8个字节又是什么呢

这8字节就是isa指针。NSObject.h头文件中声明

@interface NSObject <NSObject> 
{
    Class isa  OBJC_ISA_AVAILABILITY;
}
而Class在objc.h中是这样声明的
typedef struct objc_class *Class;

OC中所有的类都是继承自NSObject类, 所以自然也就拥有了isa指针。其后就是我们类中自己声明的成员变量了

通过以上观察,调用对象方法时传入的self参数是存储在CPU 寄存器当中的


黑马程序员 OC self指针

标签:objective c   黑马程序员   

原文地址:http://blog.csdn.net/xi_niuniu/article/details/45030761

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!