NSInvocation获取目标导致EXC_BAD_ACCESS

时间:2015-02-19 15:15:21

标签: ios objective-c exc-bad-access nsinvocation

NSInvocation我遇到了一个奇怪的问题。我在网络操作完成时将其用作返回回调。让我更详细地解释前一句:

我使用定制的网络协议,它通过TCP套接字工作,我有一个使用该协议的类,并作为我的服务器的连接。现在这个类有一个方法可以说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之外,首先不要为与请求相关的任何事情烦恼。它的工作原理如下:当我从服务器得到响应时,请求密钥会回复给我。所以它就像设置一个HTTP头字段,当你从服务器得到响应时,它会被循环回来。这样我就可以确定提出了哪个请求。 delegateInvocationMapNSMutableDictionary,它会保留我们的调用,当我们收到回复并解析出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。苹果医生说:

The receiver’s target, or nil if the receiver has no target.

如何检查接收器是否已经解除分配?这是一个巨大的痛苦。

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

2 个答案:

答案 0 :(得分:2)

NSInvocation&#39; target属性不是ARC weak引用;它被定义为assign。如果您没有对此对象的任何引用,它将被取消分配,您将开始看到EXC_BAD_ACCESS例外。

@property(assign) id target

ARC会自动将assign属性转换为unsafe_unretained而不是weak。取消分配对象时,weak属性将设置为nil; unsafe_unretained属性将继续指向内存地址,这将是垃圾。

您可以使用retainArguments方法解决此问题。

[invocation retainArguments];

来自文档:

  

如果接收方尚未这样做,则保留接收方的目标和所有对象参数,并复制其所有C字符串参数和块。

答案 1 :(得分:0)

由于NSInvocation想要保留目标,但您基本上希望它保留弱引用,请使用TPDWeakProxy之类的内容。

表示代理接受引用并使用弱指针保存它,但代理可以保持强大。

以下是我在NSInvocation类别方法中在OCMockito中的表现:

- (void)mkt_retainArgumentsWithWeakTarget
{
    if (self.argumentsRetained)
        return;
    TPDWeakProxy *proxy = [[TPDWeakProxy alloc] initWithObject:self.target];
    self.target = proxy;
    [self retainArguments];
}

这取代了目标本质上是一个弱目标。