建议您为所有 IBOutlets 声明属性以保持清晰和一致。详细信息在内存管理编程指南中有详细说明。基本要点是,当您的 NIB 对象未归档时,nib 加载代码将通过并使用 setValue:forKey: 设置所有 IBOutlets。当您在属性上声明内存管理行为时,发生的事情并不神秘。如果视图被卸载,但您使用了声明为保留的属性,您仍然可以获得对文本字段的有效引用。
也许一个更具体的例子有助于说明为什么应该使用保留属性:
我将对您工作的上下文做出一些假设——我假设上面的 UITextField 是由 UIViewController 控制的另一个视图的子视图。我会假设在某些时候,视图不在屏幕上(也许它在 UINavigationController 的上下文中使用),并且在某些时候您的应用程序会收到内存警告。
因此,假设您的 UIViewController 子类需要访问其视图以将其显示在屏幕上。此时,将加载 nib 文件,并且每个 IBOutlet 属性将由 nib 加载代码使用 setValue:forKey: 设置。这里需要注意的重要一点是将设置为 UIViewController 的 view 属性的顶级视图(将保留此顶级视图)和您的 UITextField,它也将被保留。如果它被简单地设置,它将由 nib 加载代码对其进行保留,否则该属性将保留它。UITextField 也将是顶层 UIView 的子视图,所以它上面会有一个额外的 retain,在顶层视图的 subviews 数组中,所以此时文本字段已经被保留了两次。
此时,如果您想以编程方式切换文本字段,您可以这样做。使用该属性使这里的内存管理更加清晰;您只需使用新的自动发布文本字段设置属性。如果您没有使用该属性,您必须记住释放它,并可选择保留新的。在这一点上,谁拥有这个新文本字段有点模糊,因为内存管理语义不包含在 setter 中。
现在假设一个不同的视图控制器被推送到 UINavigation 控制器的堆栈上,因此这个视图不再位于前台。在内存警告的情况下,这个离屏视图控制器的视图将被卸载。此时,顶层 UIView 的 view 属性将被清空,将被释放和释放。
因为 UITextField 被设置为保留的属性,所以 UITextField 不会被释放,因为它只会保留顶级视图的子视图数组。
相反,如果 UITextField 的实例变量没有通过属性设置,它也会存在,因为 nib 加载代码在设置实例变量时保留了它。
这里突出显示的一个有趣的点是,由于 UITextField 还通过该属性保留,因此您可能不想保留它以防出现内存警告。出于这个原因,您应该在 -[UIViewController viewDidUnload] 方法中取消该属性。这将摆脱 UITextField 上的最终版本并按预期释放它。如果使用该属性,您必须记住显式释放它。虽然这两个动作在功能上是等效的,但意图是不同的。
如果您选择从视图中删除它而不是交换文本字段,那么您可能已经从视图层次结构中删除了它并将属性设置为 nil,或者释放了文本字段。虽然在这种情况下可以编写正确的程序,但很容易在 viewDidUnload 方法中产生过度释放文本字段的错误。过度释放对象是导致崩溃的错误;将已经为 nil 的属性再次设置为 nil 不是。
我的描述可能过于冗长,但我不想在场景中遗漏任何细节。当您遇到更复杂的情况时,只需遵循指南将有助于避免问题。
另外值得注意的是,Mac OS X 在桌面上的内存管理行为有所不同。在桌面上,设置没有 setter 的 IBOutlet 不会保留实例变量;但如果可用,再次使用设置器。