gpt4 book ai didi

ios - NSInvocation 获取导致 EXC_BAD_ACCESS 的目标

转载 作者:行者123 更新时间:2023-11-28 21:50:14 27 4
gpt4 key购买 nike

我对 NSInvocation 有一个奇怪的问题。当网络操作完成时,我将它用作返回回调。让我更详细地解释一下前面的句子:

我正在使用通过 TCP 套接字工作的定制网络协议(protocol),并且我有一个使用此协议(protocol)并用作与我的服务器的连接的类。现在这个类有一个方法可以说 performNetworkRequestWithDelegate: 是这样实现的:

- (void)performNetworkRequestWithDelegate:(id<MyClassDelegate>)delegate
{
NSString *requestKey = [self randomUniqueString];
id request = [self assembleRequestAndSoOnAndSoForth];
[request setKey:requestKey];

SEL method = @selector(callbackStatusCode:response:error:);
NSMethodSignature *signature = [delegate methodSignatureForSelector:method];
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
invocation.target = delegate;
invocation.selector = method;

delegateInvocationMap[requestKey] = invocation; //See below for an explanation what the delegateInvocationMap is

[self sendRequest:request];
}

好的,所以我知道有些事情需要解释。首先,除了 requestKey 之外,不要理会与请求相关的任何事情。它是这样工作的:当我从服务器收到响应时,请求 key 会循环返回给我。因此,这就像设置一个 HTTP header 字段,当您从服务器获得响应时,该字段会返回给您。这样我就可以确定发出了哪个请求。 delegateInvocationMap 是一个 NSMutableDictionary,它保留我们的调用,当我们得到响应并解析出 requestKey 时,我们可以得到正确的调用。

现在响应的处理程序是这样的:

- (void)processResponse:(id)response
{
//Check for errors and whatnot

NSString *requestKey = [response requestKey];
if (!requestKey) return; //This never happens and is handled more correctly but keep it like this for the sake of simplicity

NSInvocation *invocation = delegateInvocationMap[requestKey];
if (!invocation) return;

[delegateInvocationMap removeObjectForKey:requestKey];

if (!invocation.target) return; //THIS LINE IS THE PROBLEM

[self setInvocationReturnParams:invocation fromResponse:response];
[invocation invoke]; //This works when everything is fine
}

当有成功的响应返回或有任何错误我正确处理时,此功能也有效。除了一个:

当调用的目标被释放时,我在尝试检查是否有我的调用目标时得到一个 EXC_BAD_ACCESS。苹果文档说:

接收者的目标,如果接收者没有目标则为 nil。

如何检查接收者是否已经被释放?这是一个巨大的痛苦。

编辑: 在下面的评论中,我发现访问已解除分配的对象始终是未知行为。我不知道是否有任何官方文档对此进行说明(我还没有检查),但我有一个解决方法。是否可以通过 KVO 观察 dealloc 调用的调用目标?

最佳答案

NSInvocationtarget 属性不是 ARC weak 引用;它被定义为 assign。如果您不持有对此对象的任何引用,它将被释放,您将开始看到 EXC_BAD_ACCESS 异常。

@property(assign) id target

ARC 自动将 assign 属性转换为 unsafe_unretained 而不是 weak。当对象被释放时,weak 属性将被设置为 nilunsafe_unretained 属性将继续指向内存地址,这将是垃圾。

您可以使用 retainArguments 方法来解决这个问题。

[invocation retainArguments];

来自文档:

If the receiver hasn’t already done so, retains the target and all object arguments of the receiver and copies all of its C-string arguments and blocks.

关于ios - NSInvocation 获取导致 EXC_BAD_ACCESS 的目标,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28609912/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com