iOS CoreData介绍和使用(以及一些注意事项)

iOS CoreData介绍和使用(以及一些注意事项)

最近花了一点时间整理了一下CoreData,对于经常使用SQLite的我来说,用这个真的有点用不惯,个人觉得实在是没发现什么亮点,不喜勿喷啊。不过这门技术的出现也有其存在价值,这是不可否认的事实,即使是不喜欢我们也得去了解一下,因为你不用别人会用,这年头都多人开发,多学点还是有好处的。废话不多说了,该开始正经事了。

CoreData介绍

CoreData是一个模型层的技术,也是一种持久化技术(数据库),它能将模型对象的状态持久化到磁盘里,我们不需要使用SQL语句就能对它进行操作。不过在性能方面是弱于直接使用SQLite的。

CoreData使用

理论我就不多说了,觉得我说的不够详细的可以自行搜索关于更多的CoreData介绍。

  • 创建步骤流程
    • 第一步先创建.xcdatamodeld文件(New File -> iOS -> Core Data ->Data Model)
iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-07 下午10.40.16.png

名字虽然可以任意取,但最好还是取和自己存储数据库名字一样的名字。这样可读性更高些。(ps:这个文件就相当于数据库文件一样,数据库文件中可以有多个表,表中可以有各个字段值,这里也可以有多个实体,每个实体有各个键值)

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-07 下午10.40.40.png

通过上面的操作就可以创建一个.xcdatamodeld文件,现在我们点击这个文件,在最下面找到Add Entity按钮,进行实体添加。

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午8.50.10.png

现在我们添加了一个名字为Entity的实体,这个名字是默认名字,现在我们可以对他进行名字的更改,对它双击一下即可修改名字,或者在xcode右边的信息栏中也可以修改

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午9.09.38.png

通过上面操作我们已经创建好了实体的名字,现在我们需要往实体中添加我们需要的键值。(ps:相当于数据库表中的字段),具体操作看图说话(实体之间的关联我就不介绍了,有兴趣的可以自行搜索资料,个人觉得会了单个实体创建外加查询就行,关联也无非是找到关联的实体*同的键值从而取得另外一个实体对象,我们可以直接先从一个实体取得指定键值对应的属性,再通过属性值去查询另一个实体,只是没关联的那么方便而已)。

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午9.17.43.png
  • 第二步创建关联类来操控CoreData实体对象(选中.xcdatamodeld文件->xcode菜单栏->Edit->Create NSManagedObject Subclass)
iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午9.35.57.png

选中自己需要关联的.xcdatamodeld文件名称,点击下一步即可。

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午9.47.24.png

选中.xcdatamodeld文件中需要关联的实体对象,点击下一步然后在选择存储目录即可。

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午9.48.06.png

完成后会发现自动生成了实体名称对应的类和扩展类(Entity.h/.m和Entity+CoreDataProperties.h/.m)

iOS CoreData介绍和使用(以及一些注意事项)
屏幕快照 2016-07-13 下午10.23.22.png

到这里就完成了整一个创建的流程,个人觉得说的有点过于详细了(导致看起来有点啰嗦)

  • 注意
    我在试验的过程中发现,如果我把关联的类从项目中删除,在对实体名称再次修改后,重新创建关联类,发现类名还是原来的实体名称显示,可能xcode没刷新,反正最后我是删除实体重新Add
    Entity之后才有用,所以为了不必要麻烦,最好还是想好实体名称后再创建关联类。
  • 关联类的理解

    我就以我自己创建的类来说明

    #import "UploadEntity.h"
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface UploadEntity (CoreDataProperties)
    
    @property (nullable, nonatomic, retain) NSString *fileName;
    @property (nullable, nonatomic, retain) NSString *fileSize;
    @property (nullable, nonatomic, retain) NSNumber *fileType;
    @property (nullable, nonatomic, retain) NSNumber *finishStatus;
    @property (nullable, nonatomic, retain) NSData *imageData;
    @property (nullable, nonatomic, retain) NSNumber *time;
    @property (nullable, nonatomic, retain) NSString *urlPath; @end NS_ASSUME_NONNULL_END
    • 该类继承NSManagedObject类

    • 可以看到在实体中的每一个键值都被xcode自动生成了实体类的对象属性,而且对于基本数据类型被自动转成了NSNumber。

    • 每个对象的修饰词都有nullable,表示CoreData数据库存储的对象可能为nil(允许存储nil,就如数据库一样,字段值也能为NULL),也就是说我们以后从CoreData读取的数据是有可能为nil。

    • 这个类对象的创建我们不能用以往的[[NSObject alloc] init]方式去创建,因为经过验证直接这样创建它,然后对属性赋值会直接抛出异常,估计内部经行了断言操作吧。苹果API提供了几个专门创建实体对象的方法,后面我在讲。

    • 因为各种限制的原因,比如创建对象限制等,导致了我不好去封装,最后不得已想到个折中办法(下面会讲)。基本这一轮下来我的兴趣缺失了很多。

  • CoreData API介绍

    • NSManagedObjectContext

       这个对象有点像SQLite对象(个人理解:用来管理.xcdatamodeld中的数据)。
      负责数据和应用库之间的交互(CRUD,即增删改查、保存等接口都是用这个对象调用).
      每个 NSManagedObjectContext 和其他 NSManagedObjectContext 都是完全独立的。
      所有的NSManagedObject(个人理解:实体数据)都存在于NSManagedObjectContext中。
      每个NSManagedObjectContext都知道自己管理着哪些NSManagedObject(个人理解:实体数据)
      // 创建方式(一般这个对象最好声明成成员变量)
      self.context = [[NSManagedObjectContext alloc] init];
      // 保存信息(比较重要的操作,每次更新和插入以及删除后必须做的操作)
      NSError *error = nil;
      BOOL result = [self.context save:&error];
      if (!result) {
      NSLog(@"添加数据失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      NSLog(@"添加数据成功");
      if (success) {
      success();
      }
      }
    • NSManagedObjectModel
      Core Data的模型文件,有点像SQLite的.sqlite文件(个人理解:表示一个.xcdatamodeld文件)
      // 创建方式
      // 1.主动加载指定名称的.xcdatamodeld资源
      //获取模型路径
      NSURL *modelURL = [[NSBundle mainBundle] URLForResource:modelName
      withExtension:@"momd"];
      //根据模型文件创建模型对象
      NSManagedObjectModel *model = [[NSManagedObjectModel alloc]
      initWithContentsOfURL:modelURL];
      // 2.从应用程序包中加载.xcdatamodeld模型文件
      NSManagedObjectModel *model = [NSManagedObjectModel mergedModelFromBundles:nil];
    • NSPersistentStoreCoordinator

      持久化存储库,CoreData的存储类型(比如SQLite数据库就是其中一种)。
      用来将对象管理部分和持久化部分捆绑在一起,负责相互之间的交流(中介一样)
      用来设置CoreData存储类型和存储路径
      对象和数据库之间的交互不需要我们来关心,苹果帮我们做好了。我们只需要面向OC开发
      // 创建方式
      /**
      注意:创建NSPersistentStoreCoordinator前必须创建NSManagedObjectModel
      个人理解:好比在一个数据库中创建表前需要设置表的所有字段,
      这里传入NSManagedObjectModel模型(.xcdatamodeld模型文件),
      感觉就是让CoreData知道所有模型中的实体中的所有键值,
      从而好创建CoreData数据库
      */
      // 以传入NSManagedObjectModel模型方式初始化持久化存储库
      NSPersistentStoreCoordinator *persistent = [[NSPersistentStoreCoordinator alloc]
      initWithManagedObjectModel:model];
      /*
      持久化存储库的类型:
      NSSQLiteStoreType SQLite数据库
      NSBinaryStoreType 二进制平面文件
      NSInMemoryStoreType 内存库,无法永久保存数据
      虽然这3种类型的性能从速度上来说都差不多,但从数据模型中保留下来的信息却不一样
      在几乎所有的情景中,都应该采用默认设置,使用SQLite作为持久化存储库
      */
      // 添加一个持久化存储库并设置类型和路径,NSSQLiteStoreType:SQLite作为存储库
      NSError *error = nil;
      // 名字最好和.xcdatamodeld文件的名字一样
      NSString *sqlPath = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,
      NSUserDomainMask, YES)
      lastObject]
      stringByAppendingPathComponent:@"xxx.sqlite"];
      [persistent addPersistentStoreWithType:NSSQLiteStoreType configuration:nil
      URL:[NSURL fileURLWithPath:sqlPath]
      options:nil
      error:&error];
      if (error) {
      NSLog(@"添加数据库失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      NSLog(@"添加数据库成功");
      // 设置上下文所要关联的持久化存储库(这一步千万不要忘记)
      self.context.persistentStoreCoordinator = self.persistent;
      if (success) {
      success();
      }
      }
    • NSEntityDescription
      用来描述实体(Entity):相当于数据库表中一组数据描述(纯属个人理解)
      通过Core Data从数据库中取出的对象,默认情况下都是NSManagedObject对象.
      NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性.
      setValue:forkey:存储属性值(属性名为key);
      valueForKey:获取属性值(属性名为key).
      每个NSManagedObject都知道自己属于哪个NSManagedObjectContext
      // 创建方式(用于插入数据使用:获得实体,改变实体各个属性值,保存后就代表插入)
      /**
      注意:不能用 alloc init方式创建
      通过传入上下文和实体名称,创建一个名称对应的实体对象
      (相当于数据库一组数据,其中含有多个字段)
      个人感觉有种先插入一个新的Entity从而获得Entity,在进行各属性赋值
      */
      NSManagedObject *newEntity = [NSEntityDescription insertNewObjectForEntityForName:
      entityName
      inManagedObjectContext:self.context];
  • CoreData的增删改查

    上面说了这么多始终没描述如何去操作它,现在先说说大概步骤。

    • insert
      1.根据Entity名称和NSManagedObjectContext获取一个新的NSManagedObject
      NSManagedObject *newEntity = [NSEntityDescription
      insertNewObjectForEntityForName:entityName
      inManagedObjectContext:self.context];
      2.根据Entity中的键值,一一对应通过setValue:forkey:给NSManagedObject对象赋值
      [newEntity setValue:value forKey:key];
      3.保存修改
      NSError *error = nil;
      BOOL result = [self.context save:&error];
    • delete
      1.通过查询(设置好查询条件)请求取得需要删除的NSManagedObject的所有集合
      2.通过for循环调用deleteObject:方法进行逐个删除(个人怀疑这个for循环会不会导致性能问题),
      暂时没发现其它删除方法。
      [self.context deleteObject:entity];
      3.保存修改
    • update
      1.通过查询(设置好查询条件)请求取得需要修改的NSManagedObject的所有集合
      2.通过for循环调用NSManagedObject对象的setValue:forkey:方法给各个属性赋值
      3.保存修改
    • read
      1.创建NSFetchRequest查询请求对象
      NSFetchRequest *request = [[NSFetchRequest alloc] init];
      2.设置需要查询的实体描述NSEntityDescription
      NSEntityDescription *desc = [NSEntityDescription entityForName:self.entityName
      inManagedObjectContext:self.context];
      request.entity = desc;
      3.设置排序顺序NSSortDescriptor对象集合(可选)
      request.sortDescriptors = descriptorArray;
      4.设置条件过滤(可选)
      NSPredicate *predicate = [NSPredicate predicateWithFormat:filterStr];
      request.predicate = predicate;
      5.执行查询请求
      NSError *error = nil;
      // NSManagedObject对象集合
      NSArray *objs = [self.context executeFetchRequest:request error:&error];
      // 查询结果数目
      NSUInteger count = [self.context countForFetchRequest:request error:&error];

      从上面可以看出如果我们仅仅是简单的对CoreData进行增删改查的操作的话,稍微费点心思的就属查询这一块。里面涉及到了两个对象NSSortDescriptor和NSPredicate,分别用于设置排序和过滤查询条件的,尤其是NSPredicate,它不仅仅用于CoreData,还被经常用于数组的过滤。具体对这两个对象的介绍我就不说了,感觉这一篇写多了。下面我直接上代码。不在多讲了。

  • CoreData个人小封装

    上面提到实体的创建因素导致了封装重用的难度,基本上新建一个.xcdatamodeld文件就得做好特意为它写一些操作接口的准备。

    • CoreDataAPI.h/.m

      //
      // CoreDataAPI.h
      // TedcallStorage
      //
      // Created by tedcall on 16/7/1.
      // Copyright © 2016年 pocket. All rights reserved.
      // #import <Foundation/Foundation.h> @interface CoreDataAPI : NSObject
      /**
      * 获取数据库存储的路径
      */
      @property (nonatomic,copy,readonly) NSString *sqlPath;
      /**
      * 获取.xcdatamodeld文件的名称
      */
      @property (nonatomic,copy,readonly) NSString *modelName;
      /**
      * 获取.xcdatamodeld文件中创建的实体的名称
      */
      @property (nonatomic,copy,readonly) NSString *entityName; /**
      * 创建CoreData数据库
      *
      * @param entityName 实体名称
      * @param modelName .xcdatamodeld文件名称(为nil则主动从程序包加载模型文件)
      * @param sqlPath 数据库存储的路径
      * @param success 成功回调
      * @param fail 失败回调
      *
      * @return 返回CoreDataAPI对象
      */
      - (instancetype)initWithCoreData:(NSString *)entityName modelName:(NSString *)modelName sqlPath:(NSString *)sqlPath success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 插入数据
      *
      * @param dict 字典中的键值对必须要与实体中的每个名字一一对应
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)insertNewEntity:(NSDictionary *)dict success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 查询数据
      *
      * @param sequenceKeys 数组高级排序(数组里存放实体中的key,顺序按自己需要的先后存放即可),实体key来排序
      * @param isAscending 是否上升排序
      * @param filterStr 过滤语句
      * @param success 成功后结果回调
      * @param fail 失败回调
      */
      - (void)readEntity:(NSArray *)sequenceKeys ascending:(BOOL)isAscending filterStr:(NSString *)filterStr success:(void(^)(NSArray *results))success fail:(void(^)(NSError *error))fail; /**
      * 删除数据
      *
      * @param entity NSManagedObject
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)deleteEntity:(NSManagedObject *)entity success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 更新数据
      *
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)updateEntity:(void(^)(void))success fail:(void(^)(NSError *error))fail;
      @end
      //
      // CoreDataAPI.m
      // TedcallStorage
      //
      // Created by tedcall on 16/7/1.
      // Copyright © 2016年 pocket. All rights reserved.
      // /**
      * 知识点:
      NSManagedObject:
      通过Core Data从数据库中取出的对象,默认情况下都是NSManagedObject对象.
      NSManagedObject的工作模式有点类似于NSDictionary对象,通过键-值对来存取所有的实体属性.
      setValue:forkey:存储属性值(属性名为key);
      valueForKey:获取属性值(属性名为key).
      每个NSManagedObject都知道自己属于哪个NSManagedObjectContext NSManagedObjectContext:
      负责数据和应用库之间的交互(CRUD,即增删改查、保存等接口都在这个对象中).
      所有的NSManagedObject都存在于NSManagedObjectContext中,所以对象和context是相关联的
      每个 context 和其他 context 都是完全独立的
      每个NSManagedObjectContext都知道自己管理着哪些NSManagedObject NSPersistentStoreCoordinator:
      添加持久化存储库,CoreData的存储类型(比如SQLite数据库就是其中一种)
      中间审查者,用来将对象图管理部分和持久化部分捆绑在一起,负责相互之间的交流(中介一样) NSManagedObjectModel:
      Core Data的模型文件 NSEntityDescription:
      用来描述实体:相当于数据库表中一组数据描述
      */ #import "CoreDataAPI.h"
      #import <CoreData/CoreData.h>
      @interface CoreDataAPI()
      /**
      * 数据模型对象
      */
      @property (nonatomic,strong) NSManagedObjectModel *model;
      /**
      * 上下文
      */
      @property (nonatomic,strong) NSManagedObjectContext *context;
      /**
      * 持久性存储区
      */
      @property (nonatomic,strong) NSPersistentStoreCoordinator *persistent;
      @end @implementation CoreDataAPI - (instancetype)initWithCoreData:(NSString *)entityName modelName:(NSString *)modelName sqlPath:(NSString *)sqlPath success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      if (self = [super init]) {
      // 断言(实体名称和存储路径是否为nil)
      // ... _entityName = entityName;
      _modelName = modelName;
      _sqlPath = sqlPath;
      // 初始化上下文
      self.context = [[NSManagedObjectContext alloc] init];
      if (modelName) {
      //获取模型路径
      NSURL *modelURL = [[NSBundle mainBundle] URLForResource:modelName withExtension:@"momd"];
      //根据模型文件创建模型对象
      self.model = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
      } else { // 从应用程序包中加载模型文件
      self.model = [NSManagedObjectModel mergedModelFromBundles:nil];
      } // 以传入模型方式初始化持久化存储库
      self.persistent = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:self.model];
      /*
      持久化存储库的类型:
      NSSQLiteStoreType SQLite数据库
      NSBinaryStoreType 二进制平面文件
      NSInMemoryStoreType 内存库,无法永久保存数据
      虽然这3种类型的性能从速度上来说都差不多,但从数据模型中保留下来的信息却不一样
      在几乎所有的情景中,都应该采用默认设置,使用SQLite作为持久化存储库
      */
      // 添加一个持久化存储库并设置类型和路径,NSSQLiteStoreType:SQLite作为存储库
      NSError *error = nil;
      [self.persistent addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:[NSURL fileURLWithPath:sqlPath] options:nil error:&error];
      if (error) {
      NSLog(@"添加数据库失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      NSLog(@"添加数据库成功");
      // 设置上下文所要关联的持久化存储库
      self.context.persistentStoreCoordinator = self.persistent;
      if (success) {
      success();
      }
      }
      } return self;
      } // 添加数据
      - (void)insertNewEntity:(NSDictionary *)dict success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      if (!dict||dict.allKeys.count == 0) return;
      // 通过传入上下文和实体名称,创建一个名称对应的实体对象(相当于数据库一组数据,其中含有多个字段)
      NSManagedObject *newEntity = [NSEntityDescription insertNewObjectForEntityForName:self.entityName inManagedObjectContext:self.context];
      // 实体对象存储属性值(相当于数据库中将一个值存入对应字段)
      for (NSString *key in [dict allKeys]) {
      [newEntity setValue:[dict objectForKey:key] forKey:key];
      }
      // 保存信息,同步数据
      NSError *error = nil;
      BOOL result = [self.context save:&error];
      if (!result) {
      NSLog(@"添加数据失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      NSLog(@"添加数据成功");
      if (success) {
      success();
      }
      }
      } // 查询数据
      - (void)readEntity:(NSArray *)sequenceKeys ascending:(BOOL)isAscending filterStr:(NSString *)filterStr success:(void(^)(NSArray *results))success fail:(void(^)(NSError *error))fail
      {
      // 1.初始化一个查询请求
      NSFetchRequest *request = [[NSFetchRequest alloc] init];
      // 2.设置要查询的实体
      NSEntityDescription *desc = [NSEntityDescription entityForName:self.entityName inManagedObjectContext:self.context];
      request.entity = desc;
      // 3.设置查询结果排序
      if (sequenceKeys&&sequenceKeys.count>0) { // 如果进行了设置排序
      NSMutableArray *array = [NSMutableArray array];
      for (NSString *key in sequenceKeys) {
      /**
      * 设置查询结果排序
      * sequenceKey:根据某个属性(相当于数据库某个字段)来排序
      * isAscending:是否升序
      */
      NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:key ascending:isAscending];
      [array addObject:sort];
      }
      if (array.count>0) {
      request.sortDescriptors = array;// 可以添加多个排序描述器,然后按顺序放进数组即可
      }
      }
      // 4.设置条件过滤
      if (filterStr) { // 如果设置了过滤语句
      NSPredicate *predicate = [NSPredicate predicateWithFormat:filterStr];
      request.predicate = predicate;
      }
      // 5.执行请求
      NSError *error = nil;
      NSArray *objs = [self.context executeFetchRequest:request error:&error]; // 获得查询数据数据集合
      if (error) {
      if (fail) {
      fail(error);
      }
      } else{
      if (success) {
      success(objs);
      }
      }
      } // 更新数据
      - (void)updateEntity:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      NSError *error = nil;
      [self.context save:&error];
      if (error) {
      NSLog(@"删除失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      if (success) {
      success();
      }
      } } // 删除数据
      - (void)deleteEntity:(NSManagedObject *)entity success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      // 传入需要删除的实体对象
      [self.context deleteObject:entity];
      // 同步到数据库
      NSError *error = nil;
      [self.context save:&error];
      if (error) {
      NSLog(@"删除失败:%@",error);
      if (fail) {
      fail(error);
      }
      } else {
      if (success) {
      success();
      }
      }
      }
      @end
    • 用法(UploadCoreDataAPI.h/.m)

      //
      // UploadCoreDataAPI.h
      // TedcallStorage
      //
      // Created by tedcall on 16/7/14.
      // Copyright © 2016年 pocket. All rights reserved.
      // #import <Foundation/Foundation.h>
      @class ResourceModel,DownLoadModel;
      @interface UploadCoreDataAPI : NSObject
      /**
      * 上传数据库模型名称
      */
      @property (nonatomic,copy,readonly) NSString *coreDataModelName;
      /**
      * 上传数据库实体名称
      */
      @property (nonatomic,copy,readonly) NSString *coreDataEntityName;
      /**
      * 上传数据库存储路径
      */
      @property (nonatomic,copy,readonly) NSString *coreDataSqlPath;
      + (instancetype)sharedInstance;
      /**
      * 插入上传记录
      *
      * @param model 数据模型
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)insertUploadModel:(ResourceModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 更新上传记录
      *
      * @param model 数据模型
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)updateUploadModel:(DownLoadModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 删除一条上传记录
      *
      * @param model 数据模型
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)deleteUploadModel:(DownLoadModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 删除所有上传记录
      *
      * @param success 成功回调
      * @param fail 失败回调
      */
      - (void)deleteAllUploadModel:(void(^)(void))success fail:(void(^)(NSError *error))fail; /**
      * 查询上传数据库所有数据
      *
      * @param success 成功回调(finishArray:已完成(DownLoadModel对象数组) unfinishedArray:未完成(DownLoadModel对象数组))
      * @param fail 失败回调
      */
      - (void)readAllUploadModel:(void(^)(NSArray *finishArray,NSArray *unfinishedArray))success fail:(void(^)(NSError *error))fail;
      @end
      //
      // UploadCoreDataAPI.m
      // TedcallStorage
      //
      // Created by tedcall on 16/7/14.
      // Copyright © 2016年 pocket. All rights reserved.
      // #import "UploadCoreDataAPI.h"
      #import "CoreDataAPI.h"
      #import "ResourceModel.h"
      #import "DownLoadModel.h"
      static NSString * const modelName = @"Upload";
      static NSString * const entityName = @"UploadEntity";
      static NSString * const sqliteName = @"Upload.sqlite";
      @interface UploadCoreDataAPI()
      @property (nonatomic,strong) CoreDataAPI *uploadData;
      @end
      @implementation UploadCoreDataAPI
      static UploadCoreDataAPI *uploadCoreData = nil;
      + (instancetype)sharedInstance
      {
      static dispatch_once_t onceToken;
      dispatch_once(&onceToken, ^{
      uploadCoreData = [[UploadCoreDataAPI alloc] init];
      }); return uploadCoreData;
      } + (instancetype)allocWithZone:(struct _NSZone *)zone
      {
      static dispatch_once_t onceToken;
      dispatch_once(&onceToken, ^{
      if (uploadCoreData == nil) {
      uploadCoreData = [super allocWithZone:zone];
      }
      }); return uploadCoreData;
      } - (instancetype)init
      {
      if (self = [super init]) {
      [self initUploadCoreData];
      }
      return self;
      } - (void)initUploadCoreData
      {
      _coreDataEntityName = entityName;
      _coreDataModelName = modelName;
      _coreDataSqlPath = [[[FileManager shardInstance] getDocumentPath] stringByAppendingPathComponent:sqliteName];
      self.uploadData = [[CoreDataAPI alloc] initWithCoreData:self.coreDataEntityName modelName:self.coreDataModelName sqlPath:self.coreDataSqlPath success:^{
      NSLog(@"initUploadCoreData success");
      } fail:^(NSError *error) {
      NSLog(@"initUploadCoreData fail");
      }]; } #pragma mark - -- 插入上传记录
      - (void)insertUploadModel:(ResourceModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      NSString *fileName = model.fileName;
      NSString *fileSize = [NSString stringWithFormat:@"%.2lf",model.size];
      NSString *urlPath = model.path;
      NSNumber *time = [NSNumber numberWithInt:[[DateManager sharedInstance] getSecondsSince1970]];
      NSNumber *fileType = [NSNumber numberWithInt:model.fileType];
      NSNumber *finishStatus = [NSNumber numberWithBool:NO];
      NSDictionary *dict = NSDictionaryOfVariableBindings(fileName,fileSize,urlPath,time,fileType,finishStatus); [self.uploadData insertNewEntity:dict success:^{
      if (success) {
      success();
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      } #pragma mark - -- 更新上传记录
      - (void)updateUploadModel:(DownLoadModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      NSString *filterStr = [NSString stringWithFormat:@"time = %d AND urlPath = '%@' AND fileName = '%@'",model.time,model.urlPath,model.fileName];
      [self.uploadData readEntity:nil ascending:YES filterStr:filterStr success:^(NSArray *results) {
      if (results.count>0) {
      NSManagedObject *obj = [results firstObject];
      [obj setValue:[NSNumber numberWithBool:model.finishStatus] forKey:@"finishStatus"];
      [self.uploadData updateEntity:^{
      if (success) {
      success();
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      } #pragma mark - -- 删除一条上传记录
      - (void)deleteUploadModel:(DownLoadModel *)model success:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      NSString *filterStr = [NSString stringWithFormat:@"time = %d AND urlPath = '%@' AND fileName = '%@'",model.time,model.urlPath,model.fileName];
      [self.uploadData readEntity:nil ascending:YES filterStr:filterStr success:^(NSArray *results) {
      if (results.count>0) {
      NSManagedObject *obj = [results firstObject];
      [self.uploadData deleteEntity:obj success:^{
      if (success) {
      success();
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      } #pragma mark - -- 删除所有上传记录
      - (void)deleteAllUploadModel:(void(^)(void))success fail:(void(^)(NSError *error))fail
      {
      [self.uploadData readEntity:nil ascending:YES filterStr:nil success:^(NSArray *results) {
      for (NSManagedObject *obj in results){
      [self.uploadData deleteEntity:obj success:^{
      if (success) {
      success();
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      } #pragma mark - -- 查询所有上传记录
      - (void)readAllUploadModel:(void(^)(NSArray *finishArray,NSArray *unfinishedArray))success fail:(void(^)(NSError *error))fail
      {
      [self.uploadData readEntity:nil ascending:YES filterStr:nil success:^(NSArray *results) {
      NSMutableArray *finishArray = [NSMutableArray array];
      NSMutableArray *unfinishedArray = [NSMutableArray array];
      for (NSManagedObject *obj in results) {
      DownLoadModel *model = [[DownLoadModel alloc] init];
      // 获取数据库中各个键值的值
      model.fileName = [obj valueForKey:@"fileName"];
      model.fileSize = [obj valueForKey:@"fileSize"];
      model.urlPath = [obj valueForKey:@"urlPath"];
      model.time = [[obj valueForKey:@"time"] intValue];
      model.fileType = [[obj valueForKey:@"fileType"] intValue];
      model.imageData = [obj valueForKey:@"imageData"];
      model.finishStatus = [[obj valueForKey:@"finishStatus"] intValue];
      if (model.finishStatus) {
      [finishArray addObject:model];
      } else {
      [unfinishedArray addObject:model];
      } }
      if (success) {
      success(finishArray,unfinishedArray);
      }
      } fail:^(NSError *error) {
      if (fail) {
      fail(error);
      }
      }];
      }
      @end

结语:以上纯属个人摸索和个人总结,不对的地方忘指出。其实在实际开发中不仅仅是增删改查这么简单,有时候会出现APP已经发布,但是数据库后续改变了,这就涉及到数据库的迁移,以及一些数据安全问题和线程等都是比较深入的,有专研精神的可以自行搜索相关资料。互勉!!!

上一篇:如何保护 Apple ID 避免被盜用的风险?


下一篇:11个好用到起飞的「Python字典」知识点