添加了如何在下面调用该方法,以及泄漏的对象是“allDBObjects”如果我删除它并将下面的字典更改为“NSMutableDictionary * objectsById”,则没有泄漏。
几天后没有这个问题我回来了。我认为这完全取决于“解除分配问题”(https://developer.apple.com/library/ios/technotes/tn2109/_index.html#//apple_ref/doc/uid/DTS40010274-CH1-SUBSECTION11)。我已经使用MKNetworkKit和AFNetworking 1.3.3进行了测试(更改1方法以使用AFNetworking而不是MKNetwork Kit)并且仍然在我的完成块中泄漏这些对象。我在块中没有引用self并且使用AFNetworking我可以看到completionBlock设置为nil并且我试图通过设置[weakOp setCompletionBlock:nil]来手动中断保留周期。
编辑:下面的代码示例我尝试使用属性并将它们引用为weakSelf。我现在已将这些更改为局部变量,但它们仍然泄漏。
有什么想法吗?
我已经接管了一个使用MKNetworkKit和Core Data的项目,在通过Leaks in instruments运行项目后,我可以在应用程序的不同位置看到很多泄漏的对象。
调试代码后,我可以看到泄漏的对象是在MKNetworkKit请求(setCompletionBlock :)的回调中发生的2次获取请求。需要完成提取请求以检查是否需要插入或更新数据。
一些进一步的信息。在完成块内部,我获取ManagedObjectContext的一个实例,并使用并发类型“NSPrivateQueueConcurrencyType”创建它,并在moc上执行插入我正确调用“performBlock:”。
请告知。
詹姆斯
示例代码: 请注意:我已经注释掉了2个获取请求,没有泄漏并将它们放回原因导致数百个对象泄漏,我设置NSDictionary和NSArray的weakSelf属性也是(非原子的,强大的)。
- (void) updateDbObjects: (int) page withCallback: (CompletionResultsNumberBlock) callback {
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
dispatch_queue_t callerQueue = dispatch_get_current_queue();
#pragma GCC diagnostic warning "-Wdeprecated-declarations"
__weak typeof(self) weakSelf = self;
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:[self createFullPath:urlStr]]];
AFJSONRequestOperation *operation = [AFJSONRequestOperation JSONRequestOperationWithRequest:request success:^(NSURLRequest *request, NSHTTPURLResponse *response, id JSON) {
NSManagedObjectContext *moc = [weakSelf managedObjectContextForCurrentThread];
DataRoot *dataRoot = [DataRoot sharedInstanceInMoc:moc];
NSArray *returnJSON = JSON[@"object"];
__block int count = returnJSON.count;
if (!count)
{
dispatch_async(callerQueue, ^{
callback(0);
});
return;
}
NSFetchRequest *fetchRequest = [NSFetchRequest fetchRequestWithEntityName:@"DBObjects"];
NSError *error;
NSArray *allDBObjects = [moc executeFetchRequest:fetchRequest error:&error];
NSMutableDictionary *objectsById = [NSMutableDictionary dictionaryWithObjects:allTeamsArray forKeys:[allTeamsArray valueForKey: GoalTeamObjectAttributes.teamId]];
for (NSDictionary *rootDict in returnJSON)
{
GoalTeamObject *dbObject = objectsById[rootDict[@"id"]];
if (dbObject == nil)
{
dbObject = [DBObjects insertInManagedObjectContext:dataRoot.managedObjectContext];
}
[weakSelf importStandardParametersFrom:rootDict into:dbObject withPrefix:@""];
}
returnJSON = nil;
objectsById = nil;
[dataRoot saveContext];
NSError *childError = nil;
if ([moc save:&childError]) {
NSError *parentError = nil;
if (![moc.parentContext save:&parentError]) {
NSLog(@"Error saving parent");
}
dispatch_async(callerQueue, ^{
callback(count);
});
} else {
NSLog(@"Error saving child");
}
} failure:^(NSURLRequest *request, NSHTTPURLResponse *response, NSError *error, id JSON) {
}];
[operation start];
}
这就是调用此代码的方式:由于存在许多数据页,因此在循环中递归调用。
__block int page = 1;
__weak typeof(self) weakSelf = self;
CompletionResultsNumberBlock loadData;
__block CompletionResultsNumberBlock block_loadData = loadData = ^(int results)
{
if (results < 100)
{
dispatch_async(callerQueue, callback);
} else {
[weakSelf updateDbObjects:++page withCallback:block_loadData];
}
};
[self updateDbObjects:page withCallback: loadData];
答案 0 :(得分:1)
这看起来不对:
__block CompletionResultsNumberBlock block_loadData = loadData = ^...
在ARC下,该块将对自己有强烈的引用。在ARC下,你应该这样做:
__block __weak CompletionResultsNumberBlock block_loadData = loadData = ^...
答案 1 :(得分:0)
您在块中的某些位置使用weakSelf
,而在其他位置使用self
。这意味着该块仍将以强有力的方式捕获self
。
我首先尝试用块中的self
替换weakSelf
的{strong>所有引用。