陈斌彬的技术博客

Stay foolish,stay hungry

Storyboard 使用 Weak Property 声明 Outlet

当我们使用 Interface Builder 生成 Outlet 对象的时候,一般都是作为 subview 来使用的。比如 UIViewController 的 view。所以说 Outlet 的持有者就是 superview 对象,即有“父子”关系。当对象间有“父子”关系时,需要使用弱引用,以避免“循环引用”。

ViewController 本身是不会作为 Outlet 的所有者的,所以使用 weak property 声明。

img

简化viewDidUnload

Outlet 都使用 weak property 声明的时候,还有一个好处,就是简化 viewDidUnload 的处理。 iOS 在系统内存不足的时候,UIViewController 会将没有表示的所有 view 做 unload 处理,即调用 viewDidUnload 接口。 所以,如果是强引用的情况下,需要释放所有权,

@property (nonatomic, strong) IBOutlet UILabel *label;
-(void)viewDidUnload {
    self.label = nil; // 取消强引用,释放所有权
    [super viewDidUnload];
}

如果没有 self.label = nil 的处理,那么 UIViewController 将不会释放 label 的所有权;结果,系统是调用了 unload,但是 subview 对象始终留在内存中。随着界面上控件的增多,内存泄露会越来越大。

如果使用的是 weak property 声明的话,结果会怎样?

@property (nonatomic, weak) IBOutlet UILabel *label;

这时,系统在 unload 时,由于 label 没有被强引用,更加 ARC 的规则,这时,label 的对象即被释放。并在释放的同时,变量自动指向 nil。

- (void)viewDidUnload {
    // 这里什么也不用管
    [super viewDidUnload];
}

不过,如果我们的 viewDidUnload 只是用来释放 Outlet 用的话,那么该函数可以不用被重载的。

什么时候要用strong property

当使用 Interface Builder 生成的第一层的 view 或者 windows 被作为 Outlet 来使用的话,那么不是不能声明为弱引用 property 的。(比如,Storyboard 的各个 scene)

理由很简单,没有被任何人强引用的对象,生成之后就会立刻被释放。

综上所述,当我们在使用 Outlet 的时候,注意不同的情况来使用 strong 还是 weak。