#import "ViewController.h"
@interface Person : NSObject
@property (nonatomic,copy)NSString *name;
-(void)print;
@end
@implementation Person
-(void)print
{
NSLog(@"%@",self.name);
}
@end
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
id cls = [Person class];
void *obj = &cls;
[(__bridge id)obj print];
// Do any additional setup after loading the view.
}
@end
答案是输出的 <ViewController: 0x7f90df907960>
NSString *str = @"123";
id cls = [Person class];
void *obj = &cls;
[(__bridge id)obj print];
会输出123
这里就两个问题
1.为啥会往上取值
首先cls本身是类对象信息,相当于一个isa指针的作用,而obj指向了cls就是持有了这个isa,而一般调用函数就是通过isa去调用,然后去method_list里面找,而方法列表也确实有这个方法,所以会调用成功,然后就是寻找name这个属性,(详见iOS栈分配空间的地址问题),而类的本质是结构体,结构体第一个值是isa,后面的属性是逐渐变大的,所以这里去找name,也是往高地址去找,所以声明了str,就会输出str的内容
2.为啥能取到值,或者说super viewDidLoad干了啥
其实前面说过super关键字,super 关键字本质会先获取一个结构体,这个结构体会传入当前类和父类指针,因为结构体是越往后的值越大,所以self在下面, 父类指针,比如说这里是UIViewController,则在上面,所以会直接访问到self,所以最终输出ViewController
iOS栈分配空间的地址问题
因为在汇编里面栈空间的划分就是给一个偏移量,先划分出固定的大小,然后可以从低地址往高地址赋值,也可以从高地址往地址赋值,注意这里说的是栈空间,也就是局部变量
举个例子:
- (void)viewDidLoad {
[super viewDidLoad];
int a = 1;
int b = 2;
int c = 3;
NSLog(@"%p --- %p --- %p",&a,&b,&c); //0x7ffee15ad22c --- 0x7ffee15ad228 --- 0x7ffee15ad224
}
就可以发现这个地址是越来越低,先声明的地址比后声明的地址要更大
|