1. ARC简介:
使用ARC之后就永远也不要自己手动去写retain, release和autorelease了,编译器会自动帮你完成这些事情。既然这些手动操作都不需要我们来写了,dealloc方法也是会自动完成的,如果没有特殊资源要手动销毁,dealloc方法实际上我们也不需要写了。自动完成?那会不会因为某些问题导致性能低下?事情证明完全不需要为这个担心,ARC已经是比较成熟的技术了,而且是在编译期间就在合适的位置自动加上内存管理代码了。iOS4以上的系统都是支持ARC的,不过要额外一提的是iOS4无法支持弱引用(weak reference)。
拿官方的举例来说一下,实现一个记录Person数据的类,现在只要这么些代码:
1
2
3
4
5
6
7
8
9
|
@
interface
Person
:
NSObject
@
property
NSString *
firstName
;
@
property
NSString *
lastName
;
@
property
NSNumber *
yearOfBirth
;
@
property
Person *
spouse
;
@
end
@
implementation
Person
@
end
|
(注意,这里的property都没有标注属性,默认都会是strong类型的,strong类型后面会介绍)
然后,你可以像下面这样实现一个方法而完全不需要管内存的问题:
1
2
3
4
5
6
7
|
-
(
void
)
contrived
{
Person *
aPerson
=
[
[
Person
alloc
]
init
]
;
[
aPerson
setFirstName
:
@
"William"
]
;
[
aPerson
setLastName
:
@
"Dudney"
]
;
[
aPerson
setYearOfBirth
:
[
[
NSNumber
alloc
]
initWithInteger
:
2011
]
]
;
NSLog
(
@
"aPerson: %@"
,
aPerson
)
;
}
|
你也可以像下面这样实现一个方法而不用担心变量被过早的释放:
1
2
3
4
5
|
-
(
void
)
takeLastNameFrom
:
(
Person *
)
person
{
NSString *
oldLastname
=
[
self
lastName
]
;
[
self
setLastName
:
[
person
lastName
]
]
;
NSLog
(
@
"Lastname changed from %@ to %@"
,
oldLastname
,
[
self
lastName
]
)
;
}
|
总体来说真的是让代码变得无比的简洁,以后应该说大部分应用使用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。如下:
1
2
3
4
5
|
// Won't work:
@
property
NSString *
newTitle
;
// Works:
@
property
(
getter
=
theNewTitle
)
NSString *
newTitle
;
|
3. ARC里新增的生命期定义
属性的特性(Property Attributes):
1
2
3
4
5
6
|
// 下面的定义相当于以前: @property(retain) MyClass *myObject;
@
property
(
strong
)
MyClass *
myObject
;
// 下面的定义类似于以前的: @property(assign) MyClass *myObject;"
// 唯一不同的是,一般myObject被释放了,属性将被置nil,而不再会成为野指针
@
property
(
weak
)
MyClass *
myObject
;
|
变量的修饰符(Variable Qualifiers):
__strong 是默认值,对象将被维持在内存。一个强引用阻止指向的对象被释放,相当于给对象retain了一次。
__weak 指定了一个弱引用,并不会阻止对象被销毁。当这个对象没有被其它地方强引用的话,它就会被销毁,并且弱引用对应的变量会被置nil。
__unsafe_unretained 这是针对iOS4使用的“弱引用”版本。为什么加上了引号,因为和真正的弱引用区别在于,一旦指向的对象被销毁了,变量也会变成野指针。所以说也看到加了unsafe字样。
__autoreleasing 用来指示通过引用传递的参数(就是非值传递),将会在return的时候被自动释放一个计数,这个一般我们不需要去用。
使用的方法举个例子如下:
1
2
|
MyClass *
__weak
myWeakReference
;
MyClass *
__unsafe_unretained
myUnsafeReference
;
|
需要注意,使用弱引用的时候必须要十分的小心,看下面的例子:
1
2
|
NSString *
__weak
string
=
[
[
NSString
alloc
]
initWithFormat
:
@
"First Name: %@"
,
[
self
firstName
]
]
;
NSLog
(
@
"string: %@"
,
string
)
;
|
没有被强引用的变量是会被立即释放的,所以这里的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里选中指定的文件后双击,就会看到编译标志的编辑框了,编辑完之后回车就好。