iOS面試程序点滴

iOS new和alloc&init的区别

2017-03-22  本文已影响105人  luckySmileBoy

简要概述

iOS创建对象的两种方式:

①UIView *myView = [[UIView alloc] init];

UIView *myView = [UIView new];

① [[ObjectName alloc] init]

② [ObjectName new]

第一种方式来创建对象时, 系统首先会给变量分配内存,然后调用init方法来进行初始化, 或者调用initWith方法来初始化;

第二种方式是第一种方式的两步的概括, 系统会直接开辟好内存, 调用init方法来初始化对象,但是只能调用init方法;

具体的区别:

1.alloc在分配内存的时候使用了zone, 它是在给对象分配内存的时候把关联的对象分配到一个相邻的区域内, 以便于调用时消耗很少的内存, 提升程序的处理速度;

2.使用new来创建对象的话, 初始化方法被固定死了, 只能使用init, 不能调用其他的initWith方法;

3.使用alloc init方法, 我们可以重写init方法, 但是如果使用new来创建的话就只能走系统的init方法了.


详细讲解 一

对于NSObject对象来说new的作用是为对象分配内存空间并使用init方法完成初始化,而与alloc&init这种显式写法来说不同的是,在分配内存空间的时候alloc相比于new来说会使用default_zone,从开放的objc运行时源码中的旧Object对象一窥NSObject的new实现

NSZone原本是用于维护一块用于对象内存分配及释放的内存池的描述信息,进程默认的NSZone是在启动的时候创建并将随后所有的对象均分配在这里,也因此在做了大量的分配及释放对象内存之后,可能会产生很多的内存碎片,在做新的内存分配的时候NSZone会试图去填补这些碎片,即从碎片中找到合适的内存区块以存放新的对象,这个查找的过程是需要时间开销的。

所以如果需要在短时间内分配大量对象,则可以创建自己的NSZone,那么在分配对象内存的时候,只需要要当前zone的末尾分配即可,相比于去已经臃肿的default zone去分配这批对象,其耗时是更少的

但斯时已逝,像上面的Managing Zones的描述所说的,iOS及osx 64位运行时已经不支持自定义的zone了

好坏已无伤大雅

所以new与alloc&init的区别只剩下是显示调用还是隐式调用init的问题了,至于那个class_getVersion()是获取class的version这个其实可以自定义,一般自定义的class的version为0,但自定义的class,就算重载init其version也还是0,调用new的时候直接会进入init。其实这份object.mm已经是GNUC的遗留文件了,在objc体系中并未用到了,可以看到其已经通过ifdef  !__OBJC2__这一步而将其定义抹去了


详细讲解 二

在实际开发中很少会用到new,一般创建对象咱们看到的全是[[className alloc] init],但是并不意味着你不会接触到new,在一些代码中还是会看到[className new],那么,他们两者之间到底有什么区别呢?我们看源码:

+new

{

id newObject = (*_alloc)((Class)self, 0);

Class metaClass = self->isa;

if(class_getVersion(metaClass) > 1)

return[newObject init];

else

returnnewObject;

}

//而 alloc init 像这样:

+ alloc

{

return(*_zoneAlloc)((Class)self, 0, malloc_default_zone());

}

- init

{

returnself;

通过源码中我们发现,[className new]基本等同于[[className alloc] init];

区别只在于alloc分配内存的时候使用了zone.

这个zone是个什么东东呢?

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

而为什么不推荐使用new?

不知大家发现了没有:如果使用new的话,初始化方法被固定死只能调用init.

而你想调用initXXX怎么办?没门儿!据说最初的设计是完全借鉴Smalltalk语法来的。

传说那个时候已经有allocFromZone:这个方法,

但是这个方法需要传个参数id myCompanion = [[TheClass allocFromZone:[self zone]] init];

这个方法像下面这样:

+ allocFromZone:(void *) z

{

return (*_zoneAlloc)((Class)self, 0, z);

}

//后来简化为下面这个:

+ alloc

{

return (*_zoneAlloc)((Class)self, 0, malloc_default_zone());

}

但是,出现个问题:这个方法只是给对象分配了内存,并没有初始化实例变量。

是不是又回到new那样的处理方式:在方法内部隐式调用init方法呢?

后来发现“显示调用总比隐式调用要好”,所以后来就把两个方法分开了。

概括来说,new和alloc/init在功能上几乎是一致的,分配内存并完成初始化。

差别在于,采用new的方式只能采用默认的init方法完成初始化,

采用alloc的方式可以用其他定制的初始化方法。

文章中有多有借鉴,只是用来作为个人备忘,如有冒犯,请联系本人,谢谢

上一篇下一篇

猜你喜欢

热点阅读