NSError和__autoreleasing

chi*_*fet 36 objective-c automatic-ref-counting

有人可以__autoreleasing在下面的示例代码块中向我解释一下目的吗?

- (void)execute:(NSError * __autoreleasing *)error {
    // do stuff, possibly assigning error if something went wrong
}
Run Code Online (Sandbox Code Playgroud)

我删除了__autoreleasing所有东西似乎仍然编译/运行正常.我开始使用obj-c后ARC,所以我从来没有真正学过/理解所有那些双下划线的东西.我已阅读ARC过渡指南,但我不完全了解他们的NSError示例.

CRD*_*CRD 74

考虑ARC如何使用变量 - 每个引用变量都有一个模式(隐式或显式):,等.这种模式让ARC知道如何处理对该变量的读写操作; 例如,对于变量读取,在写入时需要释放变量中的现有引用,然后才能将其替换为新变量.ARC需要知道任何变量的模式才能运行.

现在考虑一下自己通过引用传递的变量,例如,对于你,execute你将有一个调用:

NSError *myError = nil;
...
[someObject execute:&myError]; // pass the variable itself by reference, not the variables value
Run Code Online (Sandbox Code Playgroud)

并且execute将包含以下行的任务:

- (void)execute:(NSError * __autoreleasing *)error
{
   ...
   if (error != NULL)
      *error = [NSError ...]; // assign indirectly via the reference to a variable
   ...
}
Run Code Online (Sandbox Code Playgroud)

现在,对于那个间接赋值,ARC需要知道引用变量的模式,因此它知道如何读写.这就是__autoreleasing声明中的内容,它告诉ARC它已经传递了一个对其模式为自动释放的变量的引用,并告诉ARC如何读取和写入变量的内容.删除__autoreleasing和默认模式将被假设,在这种情况下,我建议明确肯定是好的.

自动释放模式是指变量包含了不归的参考,应读取保留在必要时,并写入可以只写.它主要用于通过引用传递的变量.

您可能会注意到,在上面的示例中,变量myError具有模式(隐式),但它通过引用传递为自动释放 - 编译器通过引入临时自动释放变量自动处理此操作,复制时保留当前引用myError,并传递临时引用作为参数execute:.在调用返回后,编译器执行从临时到的正常分配myError,这导致释放任何旧引用并保留返回的引用.

有关更多详细信息,请参阅Apple的"过渡到ARC发行说明"

跟进评论

问:是否__autoreleasing隐含设定?

答:Apple的文档并不具体,但是Clang文档说它是隐含的间接参数.如上所述,我建议明确,清晰度是一件好事.

问:这个位置是否重要?

答:是的,不是......这是一个C声明,是测验问题的内容("以下是什么声明......").限定符应该在两个星号之间,因为它是指向(对象的变量)自动释放指向对象的指针的指针,但Apple声明编译器是"宽容"而不具体说明它原谅的内容.安全地玩,把它放在正确的地方.

问:如果你没有测试errorNULL做间接转让前?

答:当然你应该在你做间接之前的某个地方.显示的代码只是一个大纲,这些细节被删除并覆盖了....然而,由于多年来已经提高了几次,我可能if已经过多了,已经添加了一个合适的版本.

  • @CRD好答案。但是您可能要指出的是,在分配给* error之前,应该检查error是否为NULL。否则,由于`[foo execute:NULL]`是完全有效的执行,并且您可以说“我不在乎该错误”,因此您的方法可能会使您的应用崩溃。 (2认同)
  • 看起来在XCode 6.1中,如果您不是显式的,则会收到警告:“类型为'NSError * __ autoreleasing *'的方法参数,没有显式所有权” (2认同)