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
- 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