OC alloc&new&init

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;

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

友情链接更多精彩内容