转载自: http://blog.youkuaiyun.com/zengconggen/article/details/38024625
在使用NSInvocation的过程中,遇到一个问题。那就是在获取返回结果后,app很快crash。示例如下:
- NSMethodSignature *methodSignature = [FlightOperations instanceMethodSignatureForSelector:NSSelectorFromString(action)];
- NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSignature];
- [invocation setTarget:fOps];
- [invocation setSelector:NSSelectorFromString(action)];
- [invocation setArgument:&flightPoint atIndex:2];
- NSArray *resultSet;
- [invocation invoke];
- [invocation getReturnValue:&resultSet];
- return resultSet;
在设置Enable Zombies后发现,是由于系统多次释放NSArray * resultSet造成的非法内存访问。
原因是在arc模式下,getReturnValue:仅仅是从invocation的返回值拷贝到指定的内存地址,如果返回值是一个NSObject对象的话,是没有处理起内存管理的。而我们在定义resultSet时使用的是__strong类型的指针对象,arc就会假设该内存块已被retain(实际没有),当resultSet出了定义域释放时,导致该crash。假如在定义之前有赋值的话,还会造成内存泄露的问题。
解决办法:
使用一个unretain的对象来获取返回值,或者 用void *指针来保存返回值,然后用__bridge来转化为OC对象。
- NSArray * __unsafe_unretained tempResultSet;
- [invocation getReturnValue:&tempResultSet];
- NSArray *resultSet = tempResultSet;
或者
- void *tempResultSet;
- [invocation getReturnValue:&tempResultSet];
- NSArray *resultSet = (__bridge NSArray *)tempResultSet;
本文详细介绍了在使用NSInvocation过程中遇到的内存泄漏问题,通过设置EnableZombies发现是由于系统多次释放NSArray* resultSet造成的非法内存访问。分析原因在于arc模式下,getReturnValue:仅仅是从invocation的返回拷贝到指定的内存地址,如果返回对象是一个NSObject对象的话,是没有处理起内存管理的。提出了解决方案,包括使用一个unretain的对象来获取返回值,或者使用void*指针来保存返回值,然后用__bridge转化为OC对象。
709

被折叠的 条评论
为什么被折叠?



