带有 ARC : Custom setter does not retain 的 Objective-C

标签 objective-c automatic-ref-counting

NOTE: This was due to a bug in some XCode beta versions that has long been fixed. This Question and answer will probably not help you if you have problems with ARC.


我正在将我的项目从手动引用计数迁移到 ARC,并且偶然发现了一个问题:我如何确保保留属性的自定义 setter 实际保留?

myClass.h 中,我声明了一个属性:@property (retain) NSDate *date。我是手动设置 __strong ivar 还是让它自动生成都没有关系。

在实现中,我当然有 @synthesize date,并实现了一个自定义 setter (或只是 download the demo Xcode project ):

- (void)setDate:(NSDate *)newDate
{
  if (allowedToSetNewDate)
  {
    date = newDate;
  }
}

这似乎没有保留日期,并且在尝试访问 myClass.date 时,当 newName 在它来自的地方(自动)释放时,给我 发送到已释放实例的消息 稍后(假设启用了 Zombie;否则,它只会静静地崩溃)。

将 setter 更改为使用 date = [newDate copy] 可以解决错误,但这并不是我真正想要的。删除自定义 setter 也可行,但显然不可取。

我在这里错过了什么?如何确保保留属性的自定义 setter 实际保留在 ARC 环境中?这似乎是一项非常基本和常见的任务,我认为我忽略了一些非常明显的事情。

(注意:这不属于任何 Apple NDA 的条款,因为 ARC 作为 LLVM 的一部分公开发布)

编辑:我创建了一个小型 Xcode 项目来演示该问题并将其上传到 github。欢迎调用download it和玩耍。我已无计可施(诚然,尽管我今天的智慧还没有达到最佳状态)。

编辑:对于这个示例项目,这个问题已经解决(见已接受的答案)。不幸的是,在我无权分享的更大的项目中,问题仍然存在。作为解决方法,我添加了具有合成 setter 的重复 strong 属性(ivars 不起作用)。新的自定义 setter 现在看起来像这样:

- (void)setDate:(NSDate *)newDate
{
  if (allowedToSetNewDate)
  {
    self.date_arcretain = newDate; //this property is only there as a workaround. ARC properly retains it, but only if the setter is synthesized
    date = newDate;
  }
}

最佳答案

这对我来说像是一个错误;你的代码应该没问题。如果您还没有这样做,请在 http://bugreport.apple.com 提交错误,并附上您的示例项目。

编辑:在进一步检查您的示例项目时,这不是错误。

示例项目中的过度释放对象不是 NSDate 实例。您可以在示例项目中完全注释掉 tc.date = now 调用,您仍然会看到相同的崩溃。事实上,你可以完全去掉 NSDate 的东西。过度释放的对象实际上是TestVC对象本身。

这是正在发生的事情。

在 iOS 4.0 中,UIWindow 有一个 rootViewController 属性。以前你只是在启动应用程序时调用 [self.window addSubview:myRootcontroller.view],现在这个变化意味着窗口实际上会引用 Root View Controller 。这对于传递旋转通知等很重要。过去,我相信 UIWindow 会在添加第一个 subview 时自动尝试设置 rootViewController(如果尚未设置),但在您的示例项目中显然没有发生。这可能是由于您创建 View 的方式所致,也可能是由于 iOS 5.0 中的更改所致。无论哪种方式,它都没有记录在案的行为,因此您不能指望它会发生。

在大多数情况下,您的应用委托(delegate)将有一个指向 Root View Controller 的 ivar。这不是严格要求的,但通常会发生这种情况。但是,在您提供的示例项目中, View Controller 不属于应用程序委托(delegate)。您也没有将它设置为窗口的 Root View Controller 。结果,在 -application:didFinishLaunchingWithOptions: 方法的末尾,没有留下对 View Controller 的强引用。您的应用程序委托(delegate)没有保留它,并且窗口本身也没有保留它。因此,ARC 将其视为局部变量(确实如此),并在方法结束时将其释放。

当然,这并没有改变您的 UIButton 仍然有一个针对您的 Controller 的操作方法这一事实。但正如文档中所述,-addTarget:action:forControlEvents: 不保留目标。所以 UIButton 有一个对你的 View Controller 的悬垂引用,现在已经被释放,因为没有人对它有强引用。因此崩溃。

解决这个问题的方法是在您的应用委托(delegate)中更改这一行:

[self.window addSubview:tc.view];

为此:

self.window.rootViewController = tc;

通过这一项更改,现在一切正常。

编辑:还要确保“ARC 迁移的预检查代码”设置未打开,因为这会导致编译器将代码视为手动管理的,并且不会插入正确的保留/发布电话。

关于带有 ARC : Custom setter does not retain 的 Objective-C,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7056235/

相关文章:

iphone - 如何在XCode(iOS SDK)中添加按钮到注释 View

ios - Xcode 6 刷新 NSUserDefaults

objective-c - Xcode4中的变量值

ios - awakeFromNib 与 drawRect 中的 CALayer

Swift,为什么这个本地变量没有被释放?

Objective-C:支持 ARC 的 iOS 照片查看器?

objective-c - 从另一个类方法更新 UI - Cocoa

objective-c - ARC下id数组成员实例

objective-c - NSArray组件JoinedByString内存泄漏

objective-c - 处理ARC中的指针对指针所有权问题