【转】CoreData以及MagicalRecord (一)

mac2022-06-30  23

先粗略的了解下CoreData中的一些核心概念

1. CoreData 的核心概念

先上两幅关键的概念图

(1)NSManagedObjectModel 托管对象模型(MOM)是描述应用程序的数据模型,这个模型包含实体(Entity),特性(Property),读取请求(FetchRequest)等。

这个MOM由实体描述对象,即NSEntityDescription实例的集合组成,实体描述对象介绍见下面第7条。

作用:添加实体的属性,建立属性之间的关系

(2)NSManagedObjectContext 托管对象上下文(MOC)参与对数据对象进行各种操作的全过程,并检测数据对象的变化,已提供对undo/redo的支持及更新绑定至数据的UI

在概念图2中,托管对象上下文(MOC)通过持久化存储协调器(PSC)从持久化存储(NSPersistentStore)中获取对象时,这些对象会形成一个临时副本在MOC中形成一个对象集合,该对象集合包含了对象以及对象彼此之间的一些关系。我们可以对这些副本进行修改,然后进行保存,然后MOC会通过 PSC 对 NSPersistentStore 进行操作,持久化存储就会发生变化。

CoreData中的NSManagedObjectModel 托管对象的数据模型(MOM),通过 MOC 进行注册。MOC有插入、删除以及更新数据模型等操作,并提供了撤销和重做的支持。

作用:插入数据,更新数据,删除数据

(3)NSPersistentStoreCoordinator 持久化存储协调器(PSC)相当于数据文件管理器,处理底层的对数据文件的读取与写入,一般我们无需与它打交道

在应用程序和外部数据存储的对象之间提供访问通道的框架对象集合统称为持久化堆栈(persistence stack)。在堆栈顶部的是托管对象上下文(MOC),在堆栈底部的是持久化对象存储(persistent object stores)。在托管对象上下文和持久化对象存储之间便是持久化存储协调器(PSC)。应用程序通过类NSPersistentStoreCoordinator的实例访问持久化对象存储。

持久化存储协调器为一或多个托管对象上下文提供一个访问接口,使其下层的多个持久化存储可以表现为单一一个聚合存储。一个托管对象上下文可以基于持久化存储协调器下的所有数据存储来创建一个对象图。持久化存储协调器只能与一个托管对象模型(MOM)相关联。

(4)NSManagedObject 托管对象(MO)数据对象 与MOC相关联

托管对象必须继承自NSManagedObject或者NSManagedObject的子类。NSManagedObject能够表述任何实体。它使用一个私有的内部存储,以维护其属性,并实现托管对象所需的所有基本行为。托管对象有一个指向实体描述的引用。NSEntityDescription 实体描述表述了实体的元数据,包括实体的名称,实体的属性和实体之间的关系。

(5)Controller 控制器 

概念图1中绿色的 Array Controller,Object Controller,Tree Controller 这些控制器,一般都是通过 control+drag 将 Managed Object Context 绑定到它们,这样我们就可以在 nib 中可视化地操作数据

(6)NSFetchRequest 获取数据请求

使用托管对象上下文来检索数据时,会创建一个获取请求(fetch request)。类似Sql查询语句的功能。

(7)NSEntityDescription 实体描述

实体描述对象提供了一个实体的元数据,包括实体名(Name),类名(ClassName),属性(Properties)以及实体属性与其他实体的一些关系(Relationships)等。

(8).xcdatamodeld

里面是.xcdatamodeld文件,用数据模型编辑器编辑,编译后为.momd或.mom文件。

我们可以选中我们的应用程序(路径类似为/Users/Childhood/Library/Application Support/iPhone Simulator/7.1/Applications/005D926F-5763-4305-97FE-AE55FE7281A4),右键显示包内容,我们看到是这样的。

我们着重理解下他们之间的协同工作关系。

 

#pragma mark - CoreDataAbout - (void)saveContext { NSError* error = nil; NSManagedObjectContext *managedObjectContext = self.managedObjectContext; if (managedObjectContext != nil) { if ([managedObjectContext hasChanges] && ![managedObjectContext save:&error]) { NSLog(@"Unresolved error :%@, %@", error, [error userInfo]); abort(); }else{ NSLog(@"save success!"); } } } - (NSURL*)applicationDocDir { NSURL* url = [[[NSFileManager defaultManager] URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask]lastObject]; NSLog(@"%@", [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0]); return url; } // 后缀为.xcdatamodeld的包,里面是.xcdatamodel文件,用数据模型编辑器编辑 // 编译后为.momd或.mom文件 - (NSManagedObjectModel*)managedObjectModel { if (_managedObjectModel != nil) { return _managedObjectModel; } NSURL* modelURL = [[NSBundle mainBundle] URLForResource:@"Journal" withExtension:@"momd"]; NSLog(@"model url: %@", [modelURL path]); self.managedObjectModel = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL]; return self.managedObjectModel; } - (NSManagedObjectContext*)managedObjectContext { if (_managedObjectContext != nil) { return _managedObjectContext; } NSPersistentStoreCoordinator* coordinator = [self persistentStoreCoordinator]; if (coordinator != nil) { self.managedObjectContext = [[NSManagedObjectContext alloc]init]; [self.managedObjectContext setPersistentStoreCoordinator:coordinator]; } return self.managedObjectContext; } - (NSPersistentStoreCoordinator*)persistentStoreCoordinator { if (_persistentStoreCoordinator != nil) { return _persistentStoreCoordinator; } NSURL* storeUrl = [[self applicationDocDir] URLByAppendingPathComponent:@"Journal.sqlite"]; NSError* error = nil; self.persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:[self managedObjectModel]]; if (![self.persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeUrl options:nil error:&error]) { NSLog(@"Error: %@, %@", error, [error userInfo]); } return self.persistentStoreCoordinator; } 应用程序先创建或读取模型文件(后缀为xcdatamodeld)生成 NSManagedObjectModel 对象。生成 NSManagedObjectContext 和 NSPersistentStoreCoordinator 对象,MOC会设置自身的持久化存储协调器(PSC),通过调用PSC来对数据文件进行读写。NSPersistentStoreCoordinator 负责从数据文件(xml,sqlite,二进制文件等)中读取数据生成 Managed Object,或保存 Managed Object 写入数据文件。NSManagedObjectContext 参与对数据进行各种操作的整个过程,它可以持有多个 Managed Object。我们通过它来监测 Managed Object。监测数据对象有两个作用:支持 undo/redo 以及数据绑定。这个类最常用Array Controller,Object Controller,Tree Controller 这些控制器一般与 NSManagedObjectContext 关联,因此我们可以通过它们在 nib 中可视化地操作数据对象

 

2. Model class

  模型有点像数据库的表结构,里面包含 Entry, 实体又包含三种 Property:Attribute(属性),RelationShip(关系), Fetched Property(读取属性)。Model class 的名字多以 "Description" 结尾。我们可以看出:模型就是描述数据类型以及其关系的。

主要的 Model class 有:

 

Model Classes Managed Object ModelNSManagedObjectModel数据模型EntityNSEntityDescription抽象数据类型,相当于数据库中的表PropertyNSPropertyDescriptionEntity 特性,相当于数据库表中的一列  > AttributeNSAttributeDescription基本数值型属性(如Int16, BOOL, Date等类型的属性)  > RelationshipNSRelationshipDescription属性之间的关系  > Fetched PropertyNSFetchedPropertyDescription查询属性(相当于数据库中的查询语句)

 

 1)Entity - NSEntityDescriptionEntity 相当于数据库中的一个表,它描述一种抽象数据类型,其对应的类为 NSManagedObject 或其子类。NSEntityDescription 常用方法:+insertNewObjectForEntityForName:inManagedObjectContext: 工厂方法,根据给定的 Entity 描述,生成相应的 NSManagedObject 对象,并插入 ManagedObjectContext 中。-managedObjectClassName 返回映射到 Entity 的 NSManagedObject 类名-attributesByName 以名字为 key, 返回 Entity 中对应的 Attributes-relationshipsByName 以名字为 key, 返回 Entity 中对应的 Relationships2)Property - NSPropertyDescriptionProperty 为 Entity 的特性,它相当于数据库表中的一列,或者 XML 文件中的 value-key 对中的 key。它可以描述实体数据(Attribute),Entity之间的关系(RelationShip),或查询属性(Fetched Property)。 > Attribute - NSAttributeDescriptionAttribute 存储基本数据,如 NSString, NSNumber or NSDate 等。它可以有默认值,也可以使用正则表达式或其他条件对其值进行限定。一个属性可以是 optional 的。  > Relationship - NSRelationshipDescription Relationship 描述 Entity,Property 之间的关系,可以是一对一,也可以是一对多的关系。  > Fetched Property - NSFetchedPropertyDescriptionFetched Property 根据查询谓词返回指定 Entity 的符合条件的数据对象。

上面说的比较抽象,举个例子来说,见图:

 

我们有一个 CocoaDataDemo.xcdatamodeld 模型文件,应用程序根据它生成一个 NSManagedObjectModel 对象,这个模型有三个 Entity,每个 Entity 又可包含 Attribute Relationship, Feteched Property 三种类型的 Property。在本例中, Author Entity 包含两个Attribute : name 和 email,它们对于的运行时类均为 NSManagedObject;还包含一个与 Post 的 Relationship;没有设置  Feteched Property。

我们通常使用 KVC 机制来访问 Property。下面来看代码:

NSManagedObjectContext * context = [[NSApp delegate] managedObjectContext]; NSManagedObject * author = nil; author = [NSEntityDescription insertNewObjectForEntityForName: @"Author" inManagedObjectContext: context]; [author setValue: @"nemo@pixar.com" forKey: @"email"]; NSLog (@"The Author's email is: %@", [author valueForKey:@"email"]);

 

在上面代码中,我们先取得 NSManagedObjectContext, 然后调用 NSEntityDescription 的方法,以 Author 为实体模型,生成对应的 NSManagedObject 对象,插入 NSManagedObjectContext 中,然后给这个对象设置特性 email 的值。

 

转载于:https://www.cnblogs.com/183damon/p/5291005.html

相关资源:JAVA上百实例源码以及开源项目
最新回复(0)