I have two classes: Profile and Config. A Profile contains an NSSet of Config objects. Both Profile and Config are NSManagedObject
subclasses.
@interface Profile : NSManagedObject
@property (nonatomic, retain) NSString * name;
@property (nonatomic, retain) NSSet *configs;
- (void)print;
@end
Here is the Config class
@interface Config : NSManagedObject
@property (nonatomic, retain) NSString * otherdata;
@property (nonatomic, retain) NSString * name;
@property (nonatomic, retain) NSMutableDictionary *myDict;
@property (nonatomic, retain) Profile *profile;
- (void)print;
@end
The dictionary myDict has NSString
*keys and values. Now when I make any changes to myDict, I call the NSManagedObject
save method and that works fine with no errors. As long as I do not kill the app, everything behaves as expected.
But when I force kill the app (either in Xcode or by double pressing the home button and killing it in the row of buttons at the bottom) and then restart the app, the data in myDict reverts to what was there before, ie the new data was not actually saved. It only appeared to be saved before I killed the app.
myDict is listed as Transformable inside the xcdatamodeld file. I tried it without specifying any NSTransformer
class. I also tried it specifying a transformer class MyDictTransformer, and in Config I added this code:
In Config.h:
@interface MyDictTransformer : NSValueTransformer
@end
In Config.m:
@implementation MyDictTransformer
+ (Class)transformedValueClass
{
return [NSMutableDictionary class];
}
+ (BOOL)allowsReverseTransformation
{
return YES;
}
- (id)transformedValue:(id)value
{
return [NSKeyedArchiver archivedDataWithRootObject:value];
}
- (id)reverseTransformedValue:(id)value
{
return [NSKeyedUnarchiver unarchiveObjectWithData:value];
}
@end
Also at top of Config.m I have this:
//
// from: https://mcmap.net/q/1920601/-core-data-not-updating-a-transformable-attribute
//
+ (void)initialize {
if (self == [Config class]) {
MyDictTransformer *transformer = [[MyDictTransformer alloc] init];
[NSValueTransformer setValueTransformer:transformer forName:@"MyDictTransformer"];
}
}
Also in AppDelegate, in both applicationDidEnterBackground
and applicationWillTerminate
I call saveContext:
- (void)saveContext
{
NSError *error = nil;
NSManagedObjectContext *managedObjectContext = self.managedObjectContext;
if (managedObjectContext != nil)
{
if ([managedObjectContext hasChanges] && ![managedObjectContext save:&error])
{
/*
Replace this implementation with code to handle the error appropriately.
abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
*/
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
abort();
}
}
}
No matter what I have tried, it simply does not save the dictionary in Config. It saves any other changes in things like config.name but not in config.myDict.
A) What am I doing wrong?
B) How can I fix this, even if I have to use some other data structure than an NSMutableDictionary
to save the data?
saveContext
after changes have been made? Do you just have onemanagedObjectContext
? – JumpoffNSDictionary
isNSCoder
compliant, so you do not need a custom transformer. You can use the standard one. It sounds like you are not really saving the data. What kind of stack are you using? Single MOC on main queue, nested child/parent, UIManagedDocument, something else? – Spittoon