Основные данные не сохраняются, когда я хочу
Я разрабатываю приложение, которое сохраняет медицинские записи. Я использую основные данные с одним контекстом и одним физическим хранилищем. Несмотря на вызов [managedObjectContext save:&error], если я извлекаю контейнер через XCode, база данных sqlite не показывает самые последние данные.
Попробовал что-то вроде предложенного здесь: Основные данные не сохраняют мои данные, но у меня не получилось. Что еще хуже, через некоторое время появляются данные: приложение может отправлять sqlite через FTP для выгрузки данных для аварийного восстановления. Бывает, что на прошлой неделе ценность работы не всегда проявляется, но через неделю обычно присутствует.
Чего я не понимаю, так это: [managedObjectContext save: & error] записывает данные НЕМЕДЛЕННО в физическое хранилище? (мое предположение было бы НЕТ). Как я могу сказать основным данным, что я хочу, чтобы мои данные были записаны на месте (при условии, что это действительно возможно)? При закрытии приложения и / или перезагрузке устройства данные ядра записываются в sqlite?
правка 1 - добавление некоторого кода [managedObjectContext save: & error] вызывается в [CommonHelper SaveData:true]; только в начале блока @try.
_FollowUp.bMI=[NSNumber numberWithFloat: [TextBMI.text floatValue]];
_FollowUp.karfnosky=[NSNumber numberWithInt: [TextKarfnosky.text intValue]];
_FollowUp.charlson=[NSNumber numberWithInt: [TextCharlson.text intValue]];
_FollowUp.alimentazione=ButtonAlimentazione.titleLabel.text;
_FollowUp.farmaci=TextFarmaci.text;
_FollowUp.dolore=ButtonDolore.titleLabel.text;
AppDelegate *appDelegate = (AppDelegate *)[[UIApplication sharedApplication] delegate];
@try {
[CommonHelper SaveData:true];
UIAlertView *alert=[[UIAlertView alloc] initWithTitle:@“Data saved” message: @“Record saved successfully” delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
float height = self.view.bounds.size.height;
float width = self.view.bounds.size.width;
lblSincro = [[UILabel alloc]initWithFrame:CGRectMake(width/2-200 ,height/2+30, 400, 20)];
lblSincro.autoresizingMask = UIViewAutoresizingFlexibleWidth;
lblSincro.font = [UIFont boldSystemFontOfSize:16.0f];
lblSincro.numberOfLines = 1;
[lblSincro setTextAlignment:NSTextAlignmentCenter];
lblSincro.backgroundColor = [UIColor clearColor];
lblSincro.textColor = [UIColor whiteColor];
lblSincro.text = @"Uploading in progress";
UIActivityIndicatorView *spinner = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleWhiteLarge] ;
spinner = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleWhiteLarge] ;
spinner.frame = CGRectMake(0, 0, width,height);
spinner.tag = 3;
UIColor *color = [[UIColor alloc] initWithRed:0.0 green:0.0 blue:0.0 alpha:0.5];
spinner.backgroundColor =color;
lblSincroDet = [[UILabel alloc]initWithFrame:CGRectMake(width/2-200 ,height/2+70, 400, 20)];
lblSincroDet.autoresizingMask = UIViewAutoresizingFlexibleWidth;
lblSincroDet.font = [UIFont boldSystemFontOfSize:16.0f];
lblSincroDet.numberOfLines = 1;
[lblSincroDet setTextAlignment:NSTextAlignmentCenter];
lblSincroDet.backgroundColor = [UIColor clearColor];
lblSincroDet.textColor = [UIColor whiteColor];
[self.view addSubview:lblSincro];
[self.view addSubview:lblSincroDet];
[self.view addSubview:spinner];
[spinner addSubview:lblSincro];
[spinner addSubview:lblSincroDet];
[spinner startAnimating];
dispatch_async(dispatch_get_global_queue( DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^(void){
SyncHelper *sync=[SyncHelper alloc];
BOOL result=sync.Sincronize;
dispatch_async(dispatch_get_main_queue(), ^(void){
[spinner stopAnimating];
[lblSincro removeFromSuperview];
[lblSincroDet removeFromSuperview];
[lblSincro removeFromSuperview];
if(result){
_SaveView = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"successSincro", @"") message:NSLocalizedString(@"successSincroMessage", @"") delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[_SaveView show];
}else{
_SaveView = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"failedSincro", @"") message:NSLocalizedString(@"failedSincroMessage", @"")
delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[_SaveView show];
}
});
});
_FollowUpLoad=_FollowUp;
_isNew=false;
}@catch (NSException *ex)
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error" message: @“Failed to save data” delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
}
2 ответа
CoreData по умолчанию использует запись в журнал записи, поэтому файл.sqlite не всегда будет содержать все ваши данные, если вы не отключите это.
Вот так...
NSDictionary *options = @{
NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES,
NSSQLitePragmasOption: @{@"journal_mode": @"DELETE"}
};
[persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:options error:&error]
Вы должны получить доступ к NSManagedObjectContext в соответствии с потоками, которые они сохраняют. Прочитайте эту ссылку.
https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/CoreData/Concurrency.html
вставьте свои данные в основную очередь
dispatch_async(dispatch_get_main_queue(), ^{
[CommonHelper SaveData:true];
});
надеюсь, что это работа.