iOS OC 对象原理探索一

alloc源码探索思路

iOS中 alloc是怎么创建对象的?init做了什么? 下面研究探索一下alloc底层实现

探索方法

方法一:

直接代码下断点 :通过control + in - objc_alloc 找到 libobjc.A.dylib

image

方法二:

符号断点: libobjc.A.dylib+[NSObject alloc]: 符号断点添加如下图:

image

方法三:

通过汇编查看,

image

image

alloc源码分析

接下来我们来分析一下alloc源码,流程如下:

image

从上图源码分析中我们可以看出,alloc主要是申请开辟内存,并初始化了一个isa属性,那么我们看下面代码:

LGTeacher  *p = [LGTeacher alloc];
NSLog(@"%lu - %lu",class_getInstanceSize([p class]),malloc_size((__bridge const void *)(p)));

打印结果如下:

image

那么为什么同一个类,会出现两个不同的打印结果呢?

源码分析如下:

image

分析得知class_getInstanceSize([p class]) 相比较于 malloc_size((__bridge const void *)(p)))少了一步if(size < 16) size = 16的判断,源码如下:

size_t instanceSize(size_t extraBytes) {
        size_t size = alignedInstanceSize() + extraBytes;
        // CF requires all objects be at least 16 bytes.
        if (size < 16) size = 16;
        return size;
    }

从上源码可以看出一个对象最少占用16字节,源码中8字节对齐,是为了让编译器更容易读取(空间换时间),防止野指针。

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;
}

由源码看出,init底层其实什么也没有做,直接返回本身self,这样做的目的是为了让开发者在设计工厂模式的时候方便重写,方便扩展。

new方法源码

new方法源码:

+ (id)new {
    return [callAlloc(self, false/*checkNil*/) init];
}

由此可以看出,[className new]基本等同于[[className alloc] init]

发布了17 篇原创文章 · 获赞 9 · 访问量 170

猜你喜欢

转载自blog.csdn.net/LiangliangSpeak/article/details/105364707