iOS从手动管理内存过渡到使用ARC

来源:互联网 发布:人人美剧app for mac 编辑:程序博客网 时间:2024/06/07 14:11

1. ARC简介:

使用ARC之后就永远也不要自己手动去写retain, release和autorelease了,编译器会自动帮你完成这些事情。既然这些手动操作都不需要我们来写了,dealloc方法也是会自动完成的,如果没有特殊资源要手动销毁,dealloc方法实际上我们也不需要写了。自动完成?那会不会因为某些问题导致性能低下?事情证明完全不需要为这个担心,ARC已经是比较成熟的技术了,而且是在编译期间就在合适的位置自动加上内存管理代码了。iOS4以上的系统都是支持ARC的,不过要额外一提的是iOS4无法支持弱引用(weak reference)。

拿官方的举例来说一下,实现一个记录Person数据的类,现在只要这么些代码:

(注意,这里的property都没有标注属性,默认都会是strong类型的,strong类型后面会介绍)

然后,你可以像下面这样实现一个方法而完全不需要管内存的问题:

你也可以像下面这样实现一个方法而不用担心变量被过早的释放:

总体来说真的是让代码变得无比的简洁,以后应该说大部分应用使用ARC是一个必然的趋势。

2. 使用ARC的强制规则

  • 不可以显式的调用dealloc([super dealloc]已经不需要了),不可以调用或者实现release, retain和autorelease。当然使用@selector去调用这些方法也是不允许的。因为前面的规定,实际上不需要在dealloc里面去释放一些ARC无法管理的资源时,完全不需要写dealloc了。针对一些Core Foundation型的对象,CFRetain和CFRelease还是可以用的。
  • NSAllocateObject和NSDeallocateObject不再可以被使用,用alloc去创建对象就好。
  • 在C结构的代码里不能再使用对象指针了,需要用ObjC类来管理你的data了。
  • id类型和void *类型不会有隐式转换了,详细的一些转换操作要参考别的章节,这里就先不赘述了。“Managing Toll-Free Bridging”
  • NSAutoreleasePool无法再用了,可以使用新的更灵活的@autoreleasepool block来替代。
  • memory zones也无法用了,因为现在已经不需要NSZone了。
  • 访问器不可以以new开头,除非你手动指定一个不是new开头的getter。如下:

3. ARC里新增的生命期定义

属性的特性(Property Attributes):

变量的修饰符(Variable Qualifiers):

__strong 是默认值,对象将被维持在内存。一个强引用阻止指向的对象被释放,相当于给对象retain了一次。
__weak 指定了一个弱引用,并不会阻止对象被销毁。当这个对象没有被其它地方强引用的话,它就会被销毁,并且弱引用对应的变量会被置nil。
__unsafe_unretained 这是针对iOS4使用的“弱引用”版本。为什么加上了引号,因为和真正的弱引用区别在于,一旦指向的对象被销毁了,变量也会变成野指针。所以说也看到加了unsafe字样。
__autoreleasing 用来指示通过引用传递的参数(就是非值传递),将会在return的时候被自动释放一个计数,这个一般我们不需要去用。

使用的方法举个例子如下:

需要注意,使用弱引用的时候必须要十分的小心,看下面的例子:

没有被强引用的变量是会被立即释放的,所以这里的string在紧接的下面一句,就已经变成nil了。

这里顺带介绍一下之前就有的一些属性特性(Property Attributes):

retain:相当于目前的strong。

assign:相当于目前的weak,主要还是用在一些元数据类型int、BOOL之类的。

copy:比较特殊,是首先为对象创建了一个副本,然后对副本拥有strong的特性。

4. 用编译标志来开启和禁用ARC

当然如果是创建新的工程只要在创建的时候勾上使用ARC的勾就好。如果是在没有开启ARC的旧工程里,可以用新的编译标志 -fobjc-arc 来为单独文件开启ARC。而如果是在已经开了ARC的工程里,可以对不需要使用ARC的指定单独文件用 -fno-objc-arc 编译标志来为其禁用ARC。编译标志的添加方法是在工程属性界面TARGETS->Build Phases->Compile Sources里选中指定的文件后双击,就会看到编译标志的编辑框了,编辑完之后回车就好。

0 0
原创粉丝点击