分配对象地址,init是用来清除地址上的数据,使其初始化
objc源码地址:https://opensource.apple.com/tarballs/objc4/
Alloc对象存储位置
- NSObject创建的对象后会生成一个对象指针,
NSObject*
,这个指针保存在栈区,通过变量的申明顺序连续排列在栈中,从栈的高地址向低地址排列 - 对象指针指向堆区,堆区存放的才是这个对象实例,
- 代码连续,指针的地址和对象的地址都很有规律的连续排列着
NSObject* person1 = [[Person alloc]init];
NSObject* person2 = [[Person alloc]init];
NSObject* person3 = [[Person alloc]init];
NSLog(@"%p %p",person1, &person1 );
NSLog(@"%p %p",person2, &person2 );
NSLog(@"%p %p",person3, &person3 );
2022-06-09 18:12:30.691748+0800 ObjecDemo[97907:11364563] 0x6000022603c0 0x7ff7b5b8a6c8
2022-06-09 18:12:30.691945+0800 ObjecDemo[97907:11364563] 0x6000022603d0 0x7ff7b5b8a6c0
2022-06-09 18:12:30.692091+0800 ObjecDemo[97907:11364563] 0x6000022603e0 0x7ff7b5b8a6b8
- &person1代表的是当前指针的地址, 而person1则表示指针对象的地址,根据日志很容易得出
- 指针变量: 存储在栈区, 高 -》 低 排列
- 对象地址: 存储在堆区, 由低到高
栈 高地址
0x7ff7b5b8a6c8
0x7ff7b5b8a6c0
0x7ff7b5b8a6b8
...
0x6000022603c0
0x6000022603d0
0x6000022603e0
堆 高地址
Alloc具体实现
根据NSObject.mm
有如下几种方式创建类,最终调用了都指向了同一个方法callAlloc
// Base class implementation of +alloc. cls is not nil.
// Calls [cls allocWithZone:nil].
id
_objc_rootAlloc(Class cls)
{
return callAlloc(cls, false/*checkNil*/, true/*allocWithZone*/);
}
// Calls [cls alloc].
id
objc_alloc(Class cls)
{
return callAlloc(cls, true/*checkNil*/, false/*allocWithZone*/);
}
// Calls [cls allocWithZone:nil].
id
objc_allocWithZone(Class cls)
{
return callAlloc(cls, true/*checkNil*/, true/*allocWithZone*/);
}
// Calls [[cls alloc] init].
id
objc_alloc_init(Class cls)
{
return [callAlloc(cls, true/*checkNil*/, false/*allocWithZone*/) init];
}
// Calls [cls new]
id
objc_opt_new(Class cls)
{
...
return [callAlloc(cls, false/*checkNil*/) init];
}
alloc执行顺序
- NSObject创建
- objc_alloc
- callAlloc(cls,xx,xx) 这里传入了类对象,根据类模版分配空间
- objc_msgSend alloc
- _objc_rootAlloc
- _objc_rootAllocWithZone
- _class_createInstanceFromZone 实现内存分配和类模版绑定
- instanceSize 计算obj需要的内存,以及内存对齐
- calloc 分配对象占有的内存,
- initInstanceIsa, 绑定类型
Tips: 字节对齐一种内存优化手段,编于以指针为单位计算和查找对象地址, 对于高位的机器上开发底层代码指定小的字节对齐方式有利于减少内存碎片,也方便指针运算