5

我想要做什么

我正在使用下面的代码将数据(历史外汇汇率)从我的后端服务器(parse.com)下载到我的应用程序的核心数据存储中。

该应用程序检查本地存储的最新可用数据,并仅从服务器获取较新的数据。如果本地还没有存储数据,它会从服务器获取所有数据。

代码的设置方式,它分批获取 100 个对象的数据,将对象保存在 Core Data 中,获取现在本地存储数据的最新最新日期(通过 using NSExpression)并获取下一批直到不再有新的对象留在服务器上 ( objects.count = 0)。

因为获取速度很慢,我决定在后台线程上运行获取和核心数据保存(使用 iOS 5 提供的新核心数据多线程模型)。

从后端服务器获取工作正常,但是......

我的问题

似乎只有那些对象被NSExpression存储在磁盘上(物理上在数据库中)的对象评估,而不是仍然在内存中并且很快将被保存的对象。因此,提取主要从磁盘(内存)中检索“旧”值。

但是,当使用以下 fetch 代码(没有NSExpressionNSDictionary作为结果类型)时,我得到了当前和正确的值:

NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:localEntityName];
NSSortDescriptor *sortDescriptor = [NSSortDescriptor sortDescriptorWithKey:@"date" ascending:YES];
request.sortDescriptors = @[sortDescriptor];
NSArray *results = [backgroundContext executeFetchRequest:request error:&error];
ForexHistory *forexHistoryItem = results.lastObject;
NSDate *lastLocalDate = forexHistoryItem.date;
NSLog(@"last local date results: %@",lastLocalDate);

下面的代码有什么问题,它使用NSExpression和字典作为 fetch resultType ?

我的问题

如何确保NSExpression查找最新的本地可用日期的 which 返回最新日期?

编码

- (void)seedForexHistoryInManagedObjectContext:(NSManagedObjectContext*)context {
  NSString* const localEntityName = @"ForexHistory";
  NSString* const parseEntityName = localEntityName;
  NSString* const parseDateIdentifier = @"date";
        
  NSExpression *keyPathExpression = [NSExpression expressionForKeyPath:@"date"];
  NSExpression *maxPeriodExpression = [NSExpression expressionForFunction:@"max:"   
             arguments:@[keyPathExpression]];
        
  NSString *expressionName = @"maxDate";
  NSExpressionDescription *expressionDescription  = [[NSExpressionDescription alloc] init];
  expressionDescription.name                      = expressionName;
  expressionDescription.expression                = maxPeriodExpression;
  expressionDescription.expressionResultType      = NSDateAttributeType;
        
  NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:localEntityName];
  request.propertiesToFetch = @[expressionDescription];
  request.resultType = NSDictionaryResultType;
    
  NSArray *currencies = @[@"AUD",@"EUR",@"NZD",@"GBP",@"BRL",@"CAD",@"CNY"];
    
  dispatch_queue_t downloadQueue;
  downloadQueue = dispatch_queue_create("download", NULL); // create serial dispatch queue
        
  NSManagedObjectContext *moc = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSPrivateQueueConcurrencyType];
  moc.parentContext = context;
  dispatch_async(downloadQueue,^{
     [moc performBlockAndWait:^{
          NSArray *objects;
             do {
               NSError *error;
               NSArray *dateResults = [moc executeFetchRequest:request error:&error];
               NSAssert(dateResults.count == 1,@"Request error!");
               NSDate *lastLocalDate = dateResults.lastObject[expressionName];
               NSLog(@"last local date results: %@",lastLocalDate);
                    
               PFQuery *query = [PFQuery queryWithClassName:parseEntityName];
               query.limit = 100;
               [query orderByAscending:parseDateIdentifier];
                    
               if (lastLocalDate) [query whereKey:parseDateIdentifier greaterThan:lastLocalDate]; // else query all
                    
                objects = [query findObjects];
                
                [objects enumerateObjectsUsingBlock:^(PFObject *obj, NSUInteger idx, BOOL *stop) {
                        
                ForexHistory *forexHistory = [NSEntityDescription    insertNewObjectForEntityForName:localEntityName
                                                                                   inManagedObjectContext:moc];
                        forexHistory.date = NULL_TO_NIL(obj[@"date"]);
                        
                        [currencies enumerateObjectsUsingBlock:^(NSString *currency, NSUInteger idx, BOOL *stop) {
                            [forexHistory setValue:NULL_TO_NIL(obj[currency]) forKey:currency.lowercaseString];
                        }];
                    }];
                    NSError *saveError = nil;
                    [moc save:&saveError];
                    
                    if (!saveError) NSLog(@"%lu forex rates saved successfully.",(unsigned long)objects.count);
                    else NSLog(@"Error when downloading historic forex rates: %@",error.localizedDescription);
    
               } while (objects.count > 0);
          }];
}

感谢您的帮助!

4

2 回答 2

2

不幸的是,这是不可能的。请参阅以下文档setIncludesPendingChanges:

特别注意事项

YES不支持 将 的值与结果类型 结合使用NSDictionaryResultType,包括聚合结果的计算(例如maxmin)。对于字典,从 fetch 返回的数组反映了持久存储中的当前状态,并且不考虑上下文中的任何未决更改、插入或删除。

于 2013-01-02T12:51:44.713 回答
0

包括待处理的更改:[request setIncludePendingChanges:YES];在您进行提取之前插入。

这可确保未保存的更改包含在提取中。

于 2012-12-23T12:07:47.883 回答