iOS OC 对象原理探索一
alloc源码探索思路
iOS中 alloc是怎么创建对象的?init做了什么? 下面研究探索一下alloc底层实现
探索方法
方法一:
直接代码下断点 :通过control + in - objc_alloc 找到 libobjc.A.dylib
方法二:
符号断点: libobjc.A.dylib
+[NSObject alloc]: 符号断点添加如下图:
方法三:
通过汇编查看,
alloc源码分析
接下来我们来分析一下alloc源码,流程如下:
从上图源码分析中我们可以看出,alloc
主要是申请开辟内存,并初始化了一个isa属性,那么我们看下面代码:
LGTeacher *p = [LGTeacher alloc];
NSLog(@"%lu - %lu",class_getInstanceSize([p class]),malloc_size((__bridge const void *)(p)));
打印结果如下:
那么为什么同一个类,会出现两个不同的打印结果呢?
源码分析如下:
分析得知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]