Я новичок в GCD, и то, что кажется простым его использованием, у меня не работает. У меня есть следующий код:
+ (void)synchronizationTimerFired:(NSTimer *)theTimer
{
if ((synchronizationUpNeededFlag) || (synchronizationDownNeededFlag))
{
if ((!synchronizationUpInProgressDepthQuantity) && (!synchronizationDownInProgressDepthQuantity))
{
dispatch_queue_t synchronizationQueue = dispatch_queue_create("synchronizationQueue",NULL);
dispatch_async(synchronizationQueue, ^(void) {
NSLog(@"Top");
...code...
...code...
...code...
NSLog(@"Bottom");
});
}
}
// Check if there is no timer, or if it is not currently valid,
// and yet if synchronization is turned on,
// then establish a repeating timer to attend to synchronization related matters.
if ((!synchronizationTimer) || (!synchronizationTimer.isValid))
{
if (synchronizationOnFlag)
{
synchronizationTimer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(synchronizationTimerFired:) userInfo:nil repeats:YES];
}
}
}
В журнале написано "Верх" и больше ничего. Код в середине не имеет бесконечного цикла — он просто никогда не выполняется до конца. Я могу поставить точки останова в коде посередине, и там будет точка, в которой выполнение программы прервется, а после которой — нет. И есть точка прямо посередине, где иногда выполнение останавливается в точке останова, а иногда нет.
Мне кажется, что очередь отправки synchronizeQueue освобождается, но я не могу вызвать dispatch_retain, потому что компилятор жалуется, что dispatch_retain нельзя использовать в ARC. Что мне не хватает?
В ответ на вопросы людей о промежуточном коде выполнение программы останавливается в этом вызове метода (представленном одной из этих строк ... кода...) на строке, которая говорит if (fetchArray.count), прокомментирован ниже .
+ (NSDate *)latestParseReceivedDownUpdatedAtDateForCoreDataEntityNameString:(NSString *)coreDataEntityNameString
{
NSDate *functionReturnValue = nil;
// Create fetchRequest
NSFetchRequest *fetchRequest = [NSFetchRequest fetchRequestWithEntityName:coreDataEntityNameString];
// Set sort descriptor
[fetchRequest setSortDescriptors:[NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:@"parseReceivedDownUpdatedAtDate" ascending:NO]]];
// We are only interested in one result
[fetchRequest setFetchLimit:1];
// Execute fetchRequest
NSError *fetchError = nil;
NSArray *fetchArray = [JBSAPPDELEGATE.managedObjectContext executeFetchRequest:fetchRequest error:&fetchError];
if (fetchArray == nil)
{
NSLog(@"Unresolved error %@, %@", fetchError, [fetchError userInfo]);
abort();
}
// If there are any records at all in our persistent store, we'll have exactly one.
// But that doesn't mean it won't be nil, as if that record has never come down from
// parse it will be a nil date on the managed object.
if (fetchArray.count) // PROGRAM EXECUTION STOPS EITHER HERE, OR JUST BEFORE HERE
{
NSManagedObject *managedObject = [fetchArray objectAtIndex:0];
functionReturnValue = [managedObject valueForKey:@"parseReceivedDownUpdatedAtDate"];
}
return functionReturnValue;
}
Я добавлю, что если я просто закомментирую вызов dispatch_async, все будет выполняться нормально. Он просто выполняется в основном потоке, чего я бы предпочел не делать.
NSLog
. - person Gabriele Petronella   schedule 03.11.2013sleep(2);
и даже добавлениеsynchronizationQueue=nil;
после блока будут выполняться нормально. Так что перепроверьте свой другой код. - person Gerd K   schedule 03.11.2013NSLog
s - person IluTov   schedule 03.11.2013-executeFetchRequest:
в фоновом режиме, вам нужен отдельный контекст. - person Gerd K   schedule 03.11.2013