Core Data学习笔记二:创建NSManagedObject对象

NSManagedObject

上篇文章Core Data学习笔记一:创建CoreDataStack创建了Data Model 文件,并且创建了一些实例Entity 如果说,可视化Data Model文件在程序中,对应了NSManagedModel对象,这些Entity在程序中,所对应的对象就是NSManagedObject (当然其实并不完全对等,可以先这么理解)

Document

NSManagedObject is a generic class that implements all the basic behavior required of a Core Data model object. It is not possible to use instances of direct subclasses of NSObject (or any other class not inheriting from NSManagedObject) with a managed object context. You may create custom subclasses of NSManagedObject, although this is not always required. If no custom logic is needed, a complete Object graph can be formed with NSManagedObject instances.

这里大概有三个要点

  • NSManagedObject 代表了Data Model中的对象
  • NSManagedObject 需要和 NSManagedObjectContext 结合使用才有意义
  • NSManagedObject 满足所有你自定义的model 对象需求,通过KVC 去访问相应的属性

生成NSManagedObject子类

虽然通过KVC就可以访问NSManagedObject所有的属性,但这样使用起来不是很方便。

在Xcode Data Model Editor 中选中你要创建NSManagedObject子类的实例,在右侧Data Model Inspector中选择Codegen

Core Data学习笔记二:创建NSManagedObject对象_第1张图片
1.png
  • Manual/None
  • Category/Extension.
  • Class Definition

Manual/None is the default, and previous behavior; in this case you should implement your own subclass or use NSManagedObject.

这个是iOS10以前的默认行为,需要我们手动通过Xcode Editor-> Create NSManagedObject SubClass... 生成ClassName+CoreDataClass 文件以及 ClassName+CoreDataGeneratedProperties 其中前者,是类的定义,以及类行为定义,后者通过category 定义了类的属性(注:Objective-C中Category定义属性不支持生成成员变量,但可以生成get set 方法,CoreData属性通过@dynamic修饰,表示,运行时生成 get set 方法)

当每次修改,data model的时候,重新通过以上方法,生成NSManagedObject对象,系统只会覆盖ClassName+CoreDataGeneratedProperties文件,而不会修改ClassName+CoreDataClass 文件。

Category/Extension generates a class extension in a file named like ClassName+CoreDataGeneratedProperties. You need to declare/implement the main class (if in Obj-C, via a header the extension can import named ClassName.h).

一般情况下,因为我们不需要修改属性Category的定义,而只需要修改类行为的定义,所以这个选项,就直接不对开发者暴露Category 文件,当你修改了DataModel的时候,就可以自动同步最新的代码

Class Definition generates subclass files named like ClassName+CoreDataClass as well as the files generated for Category/Extension.

当我们也不需要为NSManagedObject定义行为的时候,我们就可以选中这个选项,然后直接在项目里引用头文件就可以直接使用。

数据库 增删改查

增加一个数据库对象,首先,需要创建一个NSEntityDescription,前面说 Data Model Editor 里的实例并不完全和NSManagedObject对等,就是因为创建NSManagedObject对象,还需要entity相关的描述对象,这个对象才与实例对等。

NSEntityDescription *entity = [NSEntityDescription entityForName:@"Doge" inManagedObjectContext:_context];
Doge *doge = [[Doge alloc] initWithEntity:entity insertIntoManagedObjectContext:_context];
doge.name = @"xxx";
...
...
//注意,这里并没有实际把对象存入数据库,实际存入需要调用NSManagedObjectContext 的save:方法
[_context save:nil];

直接访问NSManagedObject属性的set方法,就可以修改属性,同时,也只有当调用了NSManagedObjectContext 的-save:方法后才能存入实际的存储文件中

删除调用NSManagedObjectContext-deleteObject:方法,然后调用-save:方法

 Doge *doge = ...
 [_context deleteObject:doge];
 [_context save:nil]

查 会单独写一篇学习笔记:)

Demo

添加数据

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    [self importSeedJsonSeedIfNeeded];
    return YES;
}
- (void)importSeedJsonSeedIfNeeded {
    
    NSManagedObjectContext *context = [[MTCoreDataStack sharedInstance] managedContext];
    
    NSFetchRequest * fetchRequest = [Master fetchRequest];
    fetchRequest.resultType = NSCountResultType;
    
    NSError *error;
    NSArray *results = [context executeFetchRequest:fetchRequest error:&error];
    if (!error) {
        if ([results count] > 0) {
            NSInteger masterCount = [[results objectAtIndex:0] integerValue];
            if (masterCount <= 0) {
                [self importJsonSeed];
            }
        } else {
            [self importJsonSeed];
        }
        
    }else {
        [self importJsonSeed];
    }
    
}

- (void)importJsonSeed {
    NSURL *jsonURL = [[NSBundle mainBundle] URLForResource:@"seed" withExtension:@"json"];
    
    NSError *error;
    NSData *jsonData = [NSData dataWithContentsOfURL:jsonURL options:0 error:&error];
    
    if (!error) {
        NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:jsonData options:0 error:&error];
        if (!error) {
            
            NSManagedObjectContext *context = [[MTCoreDataStack sharedInstance] managedContext];
            
            NSEntityDescription *masterEntity = [NSEntityDescription entityForName:@"Master" inManagedObjectContext:context];
            NSEntityDescription *dogeEntity = [NSEntityDescription entityForName:@"Doge" inManagedObjectContext:context];
            
            NSDictionary *masterDict = jsonDict[@"master"];
            if (![masterDict isKindOfClass:[NSDictionary class]]) {
                return;
            }
            
            Master *master = [[Master alloc] initWithEntity:masterEntity insertIntoManagedObjectContext:context];
            master.name = masterDict[@"name"];
            master.age = [masterDict[@"age"] integerValue];
        
            NSArray *doges = jsonDict[@"doges"];
            
            for (NSDictionary *dogeDict in doges) {
                if (![dogeDict isKindOfClass:[NSDictionary class]]) {
                    continue;
                }
                Doge *doge = [[Doge alloc] initWithEntity:dogeEntity insertIntoManagedObjectContext:context];
                doge.name = dogeDict[@"name"];
                doge.age = [dogeDict[@"age"] integerValue];
                doge.master = master; //因为在DataModel 里设置了 inverse, 所以master 和 doge可以相互关联上
            }
            //一定要调用 save 方法,才能真正的将文件存入数据库
            [[MTCoreDataStack sharedInstance] saveContext];
        }
    }
}

源码下载

https://github.com/mengtian-li/CoreDataStackDemo/releases/tag/v0.2

你可能感兴趣的:(Core Data学习笔记二:创建NSManagedObject对象)