OC self指针

当调用对象方法时,编译器都会默认传入一个指向本对象的指针。所以不同的对象都会调用到正确的成员变量。这个指针就是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 寄存器当中的


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值