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头字段,当你从服务器得到响应时,它会被循环回来。这样我就可以确定提出了哪个请求。 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
。苹果医生说:
The receiver’s target, or nil if the receiver has no target.
如何检查接收器是否已经解除分配?这是一个巨大的痛苦。
编辑:在下面的评论中,我发现访问已解除分配的对象始终是未知行为。我不知道是否有任何官方文件说明这一点(我还没有检查)但我有一个解决方法的想法。是否可以通过KVO观察dealloc调用的调用目标?
答案 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];
}
这取代了目标本质上是一个弱目标。