有人能帮助解决ARC和内存相关的崩溃吗?
我有一个类LTRequest
,它有一个存储块指针的成员变量:
typedef void (^APICallOKBlock)(id);
@interface LTRequest : NSObject
@property (nonatomic, strong) APICallOKBlock okCB;
@end
初始化该变量的代码:
-(void)addOKBlock:(APICallOKBlock)okBlock
{
self.okCB = okBlock;
}
如果成功,我称之为阻止:
-(void)apiCallCompletedWithResult:(id)result
{
if(self.okCB != nil)
{
self.okCB(result);
}
}
作为参考,该块看起来像这样:
APICallOKBlock okBlock = ^(id result)
{
[self handleAPICallCompleted:result];
};
它调用在调度api调用的对象中定义的函数,但是该部分并不重要。重要的是在api调用成功后调用apiCallCompletedWithResult
时会发生什么。发生的情况是,块被调用,但在请求对象LTRequest
被释放后不久应用程序崩溃(SomeAppName是我们的应用程序):
Thread 6 name: Dispatch queue: com.apple.root.low-priority
Thread 6 Crashed:
0 libobjc.A.dylib 0x300bad9c objc_release + 12
1 libobjc.A.dylib 0x300c6c00 objc_storeStrong + 24
2 SomeAppName 0x00055688 -[LTRequest .cxx_destruct] (LTRequest.m:12)
3 libobjc.A.dylib 0x300bba66 object_cxxDestructFromClass + 50
4 libobjc.A.dylib 0x300bba2c object_cxxDestruct + 8
5 libobjc.A.dylib 0x300b99ac objc_destructInstance + 20
我尝试使用unsafe_unretained
而不是strong
将块存储在LTRequest
对象中:
@property (nonatomic, unsafe_unretained) APICallOKBlock okCB;
在这种情况下,崩溃的发生与if完全一致:
Thread 0 name: Dispatch queue: com.apple.main-thread
Thread 0 Crashed:
0 libobjc.A.dylib 0x300b9eda objc_retain + 10
1 libobjc.A.dylib 0x300c6bde objc_retainAutoreleasedReturnValue + 34
2 SomeAppName 0x00054f90 -[LTRequest apiCallCompletedWithResult:] (LTRequest.m:84)
3 SomeAppName 0x0002f8aa __29-[LTServerProxy makeAPICall:]_block_invoke_0 (LTServerProxy.m:154)
// file LTRequest.m
line 84 if(self.okCB != nil)
line 85 {
line 86 self.okCB(result);
line 87 }
我是否正确地使用了指向块的指针?我应该如何调用它以避免任何崩溃?
一些信息:我使用的是iOS SDK 5.0,部署目标是4.0,设备是iPhone,整个项目都启用了ARC。
您希望属性为copy
而不是retain
:
@property (nonatomic, copy) APICallOKBlock okCB;
在将块的实例保存到类实例时,需要复制块。
-(void)addOKBlock:(APICallOKBlock)okBlock
{
self.okCB = [okBlock copy];
}