iOS底层探究-浅谈alloc,init,new

iOS中alloc与init的作用

代码演示

如代码所示:

Person *p1 = [Person alloc];
Person *p2 = [p1 init];
Person *p3 = [p1 init];
NSLog(@"�%@ - %p",p1,&p1);
NSLog(@"�%@ - %p",p1,&p2);
NSLog(@"�%@ - %p",p1,&p3);

猜测:
三次打印,p1的值是否相同?&p1,&p2,&p3是否相同?

来看下打印结果:

�<Person: 0x280ebc0e0> - 0x16d965c48
�<Person: 0x280ebc0e0> - 0x16d965c40
�<Person: 0x280ebc0e0> - 0x16d965c38

我们再使用内存查看指令看下三个指针里面存放了什么数据:

(lldb) x/2xg 0x16d965c48
0x16d965c48: 0x0000000280ebc0e0 0x0000000102b07c70
(lldb) x/2xg 0x16d965c40
0x16d965c40: 0x0000000280ebc0e0 0x0000000280ebc0e0
(lldb) x/2xg 0x16d965c38
0x16d965c38: 0x0000000280ebc0e0 0x0000000280ebc0e0

结论:显而易见,p1,p2,p3 三个指针中存放的都是Person对象的内存地址,也就是说三个指针都指向了Person对象。

那么apple底层alloc和init究竟是怎样处理的,一个类的创建需要经历那些流程呢?

一:alloc的实现流程

如下图所示:

alloc 流程示意

**由流程图可以看出,alloc方法内部主要是为对象申请开辟内存空间 **

留意下_class_createInstanceFromZone 调用的这个函数:

 size_t size = cls->instanceSize(extraBytes);

内部实现

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;
    }
uint32_t unalignedInstanceSize() {
        assert(isRealized());
        return data()->ro->instanceSize;
    }
uint32_t alignedInstanceSize() {
        return word_align(unalignedInstanceSize());
    }

其中 word_align() 这个函数的实现比较有意思:

static inline uint32_t word_align(uint32_t x) {
    return (x + WORD_MASK) & ~WORD_MASK;
}

这个实现具体是什么意思呢?
WORD_MASK 是个宏定义等于7 x想当于初始化变量所占空间,两者相加的结果 再 & ~ WORD_MASK 也就是相加的值再抹去二进制的最后三位 (0 ~ 7的范围) 保证最后得到的结果是8字节对齐,不足8字节也会分配8字节的空间。

结论:
由于父类NSObject 存在一个isa成员变量,所以开辟的时候父类isa指针的内存空间也被分配并初始化出来了,指针占8个字节, 小于size = 16 ,最终分配的size 为 16。

注:为什么 size < 16 的时候也要分配16个字节的内存呢?
虽然一个对象的分配是以8字节对齐,而系统分配内存大小是以16字节对齐的,一个类对象最少占16个字节,是为了让编译器容易读取地址(空间换时间,内存距离比较规则),且防止野指针。

二: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方法的实现则比较简单,直接返回的alloc对象self本身,这个方法主要为了开发人员能够在工厂设计开发的时候容易扩展(也是本文开始两次init返回同一结果的原因)

三:new的实现

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

通过源码中我们发现,[className new]基本等同于[[className alloc] init],其中区别在于以下两点:

1:alloc分配内存的时候使用了zone,而new没有使用。

zone这个方式是给对象分配内存的时候,把关联的对象分配到一个相邻的内存区域内,以便于调用时消耗很少的代价,提升了程序处理速度;

2:如果使用new的话,初始化方法被固定死只能调用init,不方便自己去扩展定制。

第一次写,欢迎大牛提出意见,一起交流和进步,谢谢~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,670评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,928评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,926评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,238评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,112评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,138评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,545评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,232评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,496评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,596评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,369评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,226评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,600评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,906评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,185评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,516评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,721评论 2 335