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]

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值