Fuck iOS EveryDay

关于NSInvocation调用getReturnValue发生

2019-05-06  本文已影响51人  一线搬砖工人

写在前面

最近在写IQService这个模块间通信框架的时候,遇到了一个诡异的问题。虽然现在已经完美解决了,但是本着知其然且知其所以然的原则,现在把具体问题以及问题的解决方案还有背后的原理一一梳理出来。

问题描述

写完IQService核心功能,把IQServiceDemo运行起来之后,程序直接挂掉了。程序崩在main函数里面,EXC_BAD_ACCESS野指针错误,而且控制台没有给出任何输出。

tip1_1.jpeg

解决问题

EXC_BAD_ACCESS错误我们知道是访问了野指针,我们一般可以通过打开XCode->Edit Scheme->开启Zomie Objects来监控捕捉野指针地址。如下图。

tip1_2.jpeg

打开Zomie Objects之后,重新run,发现控制台输入了所访问异常的野指针地址。具体报错信息为-[CFString release]: message sent to deallocated instance 0x600003712d40。

tip1_3.jpeg

这时候我似乎发现了问题所在,在IQService中,我利用NSInvocation进行了方法的调用,并通过调用NSInvocation的getReturnValue函数获取返回值,其返回值恰好是个NSString,难道NSString被提前释放了?

tip1_4.jpeg
    /*发生问题的代码*/
    id returnValue = nil;
    
    NSInteger index = 2;
    for (id arg in argsArray) {
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wincompatible-pointer-types-discards-qualifiers"
        [invocation setArgument:&arg atIndex:index];
#pragma clang diagnostic pop
        index++;
    }
    [invocation invokeWithTarget:instance];
    if (mSignature.methodReturnLength) {
        [invocation getReturnValue:&returnValue];
        NSLog(@"%p",returnValue);
    }
    return returnValue;

问题本质

解决方案

利用__autoreleasing id returnValue = nil;即可解决此问题。这里相当于延长了returnValue所指向对象的释放时机。

上一篇下一篇

猜你喜欢

热点阅读