alloc底层调用
[Person alloc]
汇编调试 alloc -> objc_alloc
fixupMessageRef(message_ref_t *msg)
{
...
if (msg->imp == &objc_msgSend_fixup) {
if(msg->sel==@selector(alloc)) {
msg->imp= (IMP)&objc_alloc;
}
}
....
}
调用alloc 方法,会查找objc_alloc的imp
查看源码 objc_alloc -> callAlloc
callAlloc有 3个分支
第一种
if(fastpath(!cls->ISA()->hasCustomAWZ())) {
return _objc_rootAllocWithZone(cls, nil);
}
第二种
if(allocWithZone) {
return((id(*)(id,SEL,struct_NSZone*))objc_msgSend)(cls,@selector(allocWithZone:),nil);
}
第三种
return((id(*)(id,SEL))objc_msgSend)(cls,@selector(alloc));
断点调试 第一次先执行 objc_msgSend即 NSObject 的 alloc 方法
源码查看NSObject 的 alloc 方法 -> _objc_rootAlloc -> callAlloc
再一次调用callAlloc, 断点得知这次执行第一种
判断 isa 的 hasCustomAWZ()值
if(fastpath(!cls->ISA()->hasCustomAWZ()))
bool hasCustomAWZ() const {
return !cache.getBit(FAST_CACHE_HAS_DEFAULT_AWZ);
}
isa cache 缓存存在,执行_objc_rootAllocWithZone, 断点也可以得出结论
查看源码 -> _class_createInstanceFromZone
结论:alloc-> objc_alloc —> callAlloc —> objc_msgSend —> alloc —> _objc_rootAlloc —> callAlloc —> _objc_rootAllocWithZone —> _class_createInstanceFromZone
_class_createInstanceFromZone 源码中包括下面代码
实例对象 的实际大小, 8字节对齐,最小是16字节
算法 8的倍数 (x + 7) & ~7
size = cls->instanceSize(extraBytes);
define WORD_MASK7UL
static inline uint32_t word_align(uint32_t x) {
return (x + WORD_MASK) & ~WORD_MASK;
}
系统为实例对象 开辟的内存大小,以16字节对齐
obj = (id)calloc(1, size);
为什么要字节对⻬?
字节是内存的容量单位。但是,CPU在读取内存的时候,却不是以字节为单位来读取的,⽽是以“块”为单位读取的,所以⼤家也经常听到⼀块内存,“块”的⼤⼩也就是内存存取的⼒度。如果不对⻬的话,在我们频繁的存取内存的时候,CPU就需要花费⼤量的精⼒去分辨你要读取多少字节,这就会造成CPU的效率低下,如果想要CPU能够⾼效读取数据,那就需要找⼀个规范,这个规范就是字节对⻬。
为什么对象内部的成员变量是以8字节对⻬,系统实际分配的内存以16字节对⻬?
以空间换时间。苹果采取16字节对⻬,是因为OC的对象中,第⼀位叫isa指针,它是必然存在的,⽽且它就占了8位字节,就算对象中没有其他的属性了,也⼀定有⼀个isa,那对象就⾄少要占⽤8位字节。如果以8位字节对⻬的话,如果连续的两块内存都是没有属性的对象,那么它们的内存空间就会完全的挨在⼀起,是容易混乱的。以16字节为⼀块,这就保证了CPU在读取的时候,按照块读取就可以,效率更⾼,同时还不容易混乱。
new : alloc + init 的语法糖
+ (id)new{
return [callAlloc(self, false/*checkNil*/) init];
}
init : 返回创建的实例对象
工厂模式创建对象,可以重写init 方法, 统一规范
- (id)init{
return _objc_rootInit(self);
}
id _objc_rootInit(id obj)
{
// In practice, it will be hard to rely on this function.
// Many classes do not properly chain -init calls.
return obj;
}