- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
在 ARC 下,输出参数采用以下形式(默认情况下;这等同于 NSError **
):
- (BOOL)tryWithError:(NSError *__autoreleasing *)err;
来自Transitioning to ARC Release Notes ,如果我们传递一个 __strong
局部变量的地址,编译器将创建一个临时变量并生成以下代码:
NSError *error; // strong
BOOL ok = [myObject tryWithError:&error];
// translated to
NSError *__strong error;
NSError *__autoreleasing tmp = error;
BOOL ok = [myObject tryWithError:&tmp];
error = tmp;
但是如果我们用一个实例变量来做:
@implementation Foo {
NSError *_error; // strong
}
- (void)bar
{
[myObject tryWithError:&_error];
}
...
这给了我们错误
Passing address of non-local object to
__autoreleasing
parameter for write-back.
为什么这是无效的?编译器不能自动将这样的代码翻译成这个吗?
- (void)bar
{
NSError *__autoreleasing tmp = _error;
[myObject tryWithError:&tmp];
_error = tmp;
}
毕竟,这就是我要写的解决问题的方法!
注意:在参数类型will reduce the compiler's work slightly中加入out
关键字因为它不必将当前值读入临时变量——但这不会处理错误。
最佳答案
不能将指向 ivar 的指针传递给 ARC 下的“id __autoreleasing *”参数,因为这种传递回写的格式不正确。 respective section in the ARC specification列出了 pass-by-writeback 的合法形式,这里唯一适用的是
&var, where var is a scalar variable of automatic storage duration with retainable object
,因此只允许自动存储持续时间(局部变量)。
为什么这是无效的:我很确定这里的原因是与旧代码的兼容性:
1) You should only look at the error writeback in the failure case .在成功的情况下,根本无法保证错误指针中的内容。
2) 一般来说,是否应该使用回写值取决于方法的约定。这是编译器无法检查的内容。
这是与 &error
类型匹配的代码版本( NSError * __autoreleasing *
) 到写回的类型 ( NSError **
被解释为 NSError * __autoreleasing *
)。如果ok
是YES,错误值不会被触及。
NSError * __autoreleasing error;
BOOL OK = [myObject performOperationWithError:&error];
if (!OK) {
// use error
}
然而,那些__autoreleasing
很丑,所以不要强制我们使用 __autoreleasing
在所有地方,编译器允许我们传递 __strong
(但本地)变量以及(默认所有权):
NSError *error;
BOOL OK = [myObject performOperationWithError:&error];
if (!OK) {
// use error
}
根据文档,它被重写为:
NSError * __strong error;
NSError * __autoreleasing tmp = error;
BOOL OK = [myObject performOperationWithError:&tmp];
error = tmp;
if (!OK) {
// use error
}
完全没有问题,错误只会在成功的情况下使用。
现在让我们看一下 __strong
实例变量 _error
.为什么编译器不允许这样做?这是重写的样子:
NSError * __autoreleasing tmp = _error;
BOOL OK = [myObject performOperationWithError:&tmp];
_error = tmp;
if (!OK) {
// use error
}
这里的问题是 tmp
中的回写将总是被使用(分配给实例变量 _error
),忽略方法的约定回写应该只在错误情况下使用(或一般情况下任何情况)该方法的文档说)。将最后一个错误分配给实例变量的安全方法是
NSError * __autoreleasing tmp = _error;
BOOL OK = [myObject performOperationWithError:&tmp];
if (!OK) {
_error = tmp;
// use error
} else {
_error = nil; // Make sure that _error is nil if there was no error.
}
这仅适用于返回错误的 Cocoa 方法的约定。通常,编译器无法判断方法将如何处理 id *
。 : 那里可能有使用不同约定的旧方法。因此,如果您真的想将写回存储在 __strong
中实例变量,您目前必须自己多走一英里,我不希望这会改变。
关于objective-c - 为什么不能将 ivar 的地址传递给 ARC 下的 "id __autoreleasing *"参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14195017/
我在返回错误的 Apple 函数中注意到了这种模式 error:(NSError *__autoreleasing *)outError 我理解它的意思,它是指向指针的指针,用于执行结果(仅使用 *
在 AFNetworking 中我找到了这个函数: - (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request
我有一个街区; typedef void (^SIResponseHandler) (id obj, NSString *error); 和方法: + (void)uploadPhoto:(UIIma
如何使变量的地址成为 NSObject 中的强属性?我有一个名为 SCPFMessageThreadQuery 的类,它是通过传递 SCPFMessageThread 变量的地址来分配初始化的。 @i
所以我像往常一样将一个指向 NSError 对象的指针传递给一个方法。它曾经很简单: -(BOOL)foo:(NSString *)string error:(NSError **)error 但是对
我拼命想了解 __autoreleasing 关键字在 Objective-C 中的用法。我已仔细阅读以下问题的答案: In which situations do we need to write
根据 Transitioning to ARC Release Notes : __autoreleasing is used to denote arguments that are passed
有人可以向我解释在以下示例代码块中使用 __autoreleasing 的目的吗? - (void)execute:(NSError * __autoreleasing *)error { /
我一直都知道这种模式是管理 NSError** 参数(特别是 BOOL 返回值和 NSError* __autoreleasing * 参数: -(BOOL)doSomethingWithString
我正在尝试将旧的非 ARC 项目转换为 ARC,但出现此编译错误:“无法在 block 中捕获 __autoreleasing 变量” - (void)animateViewController:(_
-(NSData *)jsonRepresentation:(NSError **error)error { NSDictionary *dict = [self getDictRepresen
考虑这段代码: @implementation MyClass -(void)dealloc { NSLog(@"MyClass dealloc: %@", self); } @end @im
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: In which situations do we need to write the __autorele
根据 Cocoa 的“创建规则”,返回新实例化对象的方法必须以 new、create 或 copy 开头,如果所有权转移给调用者。 假设我有一个实现名为 (NSObject *)getCorrectO
在学习 NSInvocations 时,我对内存管理的理解似乎存在差距。 这是一个示例项目: @interface DoNothing : NSObject @property (nonatomic,
我正在尝试完成这个谜题。 __strong 是所有 Objective-C 可保留对象指针(如 NSObject、NSString 等)的默认值。它是一个强引用。 ARC 在作用域末尾用 -relea
为什么 Apple 的方法没有将 NSError** 声明为 NSError * __autoreleasing * ? Transitioning to ARC Release Notes似乎暗示他
我正在尝试从 id __autoreleasing * 转换为 CFTypeRef * (void **)。 我试过: id __autoreleasing *arg = [OCMArg setTo:
在 ARC 下,输出参数采用以下形式(默认情况下;这等同于 NSError **): - (BOOL)tryWithError:(NSError *__autoreleasing *)err; 来自T
我正在尝试使用 PowerPlot 库绘制条形图。因此,我发现只有这个代码可以满足我对负条以及阴影或不透明度的要求。原始示例项目不使用 ARC。但是我的项目使用 ARC。在复制相同的代码时,我在编译项
我是一名优秀的程序员,十分优秀!