美文网首页iOScoredata提升
Core Data学习笔记二:创建NSManagedObject

Core Data学习笔记二:创建NSManagedObject

作者: 天天有写不完的代码 | 来源:发表于2017-08-19 20:05 被阅读96次

    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

    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

        本文链接:https://www.haomeiwen.com/subject/srfxdxtx.html