iOS CoreData (一) 增删改查

作者: 且行且珍惜_iOS | 来源:发表于2017-07-17 13:09 被阅读6213次

    Core Data是iOS5之后才出现的一个框架,本质上是对SQLite的一个封装,它提供了对象-关系映射(ORM)的功能,即能够将OC对象转化成数据,保存在SQLite数据库文件中,也能够将保存在数据库中的数据还原成OC对象,通过CoreData管理应用程序的数据模型,可以极大程度减少需要编写的代码数量!

    更新于2018.3.7 iOS CoreData (二) 版本升级和数据库迁移

    增删改查.gif

    1、首先创建一个coreData 模型文件:系统创建或者自己创建

    系统创建模型文件.png 自己创建模型文件.png

    2、在data Model 中创建项目中需要用到的实体(Entities),例如,创建一个Student 实体(第一字母必须是大写),以及添加一些name、age、sex 等属性,如下图

    创建实体.png

    3、生成对应实体的实体类,在此之前要注意下图两个设置部分,否则会引起崩溃现象

    注意1.png 注意2.png 创建实体类.png 创建实体类.gif 实体类的四个文件.png

    4、生成上下文 关联数据库

    • NSManagedObjectContext 管理对象,上下文,持久性存储模型对象,处理数据与应用的交互
    • NSManagedObjectModel 被管理的数据模型,数据结构
    • NSPersistentStoreCoordinator 添加数据库,设置数据存储的名字,位置,存储方式
    • NSManagedObject 被管理的数据记录
    • NSFetchRequest 数据请求
    • NSEntityDescription 表格实体结构
    ①、自己创建模型文件时需要以下代码来手动生成上下文,关联数据库
    //创建数据库
    - (void)createSqlite{
        
        //1、创建模型对象
        //获取模型路径
        NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"Model" withExtension:@"momd"];
        //根据模型文件创建模型对象
        NSManagedObjectModel *model = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
        
        //2、创建持久化存储助理:数据库
        //利用模型对象创建助理对象
        NSPersistentStoreCoordinator *store = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:model];
        
        //数据库的名称和路径
        NSString *docStr = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
        NSString *sqlPath = [docStr stringByAppendingPathComponent:@"coreData.sqlite"];
        NSLog(@"数据库 path = %@", sqlPath);
        NSURL *sqlUrl = [NSURL fileURLWithPath:sqlPath];
        
        NSError *error = nil;
        //设置数据库相关信息 添加一个持久化存储库并设置类型和路径,NSSQLiteStoreType:SQLite作为存储库
        [store addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:sqlUrl options:nil error:&error];
        
        if (error) {
            NSLog(@"添加数据库失败:%@",error);
        } else {
            NSLog(@"添加数据库成功");
        }
        
        //3、创建上下文 保存信息 对数据库进行操作
        NSManagedObjectContext *context = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
       
        //关联持久化助理
        context.persistentStoreCoordinator = store;
        _context = context;
    
    }
    
    ②、系统创建模型文件时会自动生成关联数据库的代码,在iOS10以下和iOS10之后生成的不一样,出现了一个新类NSPersistentContainer。

    NSPersistentContainer是一个容器,封装了应用程序中的CoreData Stack(核心数据栈堆),简化了创建和管理的核心堆栈的数据处理创建NSManagedObjectModel,NSPersistentStoreCoordinator,NSManagedObjectContext。详情可以看看这篇文章:http://blog.csdn.net/u013263917/article/details/53277708

    iOS10以下系统创建模型文件时自动生成关联数据库的代码(转摘).jpeg iOS10之后系统自动生成的广联数据库的代码.png NSPersistentContainer.png
    
        AppDelegate * appDelegate = (AppDelegate *)[UIApplication sharedApplication].delegate;
        
        NSPersistentContainer * container = appDelegate.persistentContainer;
        
        //返回沙盒中存储数据库的文件夹URL路径,这是一个静态方法,表示数据库的文件路径是唯一的
        NSURL * url = [NSPersistentContainer defaultDirectoryURL];
        
        NSManagedObjectContext *viewContext = container.viewContext;
        
        NSManagedObjectModel *managedObjectModel = container.managedObjectModel;
        
        NSPersistentStoreCoordinator *persistentStoreCoordinator = container.persistentStoreCoordinator;
    
        //使用存储调度器快速在多线程中操作数据库,效率非常高(比主线程操作块50倍!!!)
        - (void)performBackgroundTask:(void (^)(NSManagedObjectContext *))block;
    
    

    5.增删改查排

    • 写入数据
      // 1.根据Entity名称和NSManagedObjectContext获取一个新的继承于NSManagedObject的子类Student
        Student * student = [NSEntityDescription  insertNewObjectForEntityForName:@"Student"  inManagedObjectContext:_context];
        
        //2.根据表Student中的键值,给NSManagedObject对象赋值
        student.name = [NSString stringWithFormat:@"Mr-%d",arc4random()%100];
        student.age = arc4random()%20;
        student.sex = arc4random()%2 == 0 ?  @"美女" : @"帅哥" ;
        student.height = arc4random()%180;
        student.number = arc4random()%100
    
        //   3.保存插入的数据
        NSError *error = nil;
        if ([_context save:&error]) {
            [self alertViewWithMessage:@"数据插入到数据库成功"];
        }else{
            [self alertViewWithMessage:[NSString stringWithFormat:@"数据插入到数据库失败, %@",error]];
        }
    
    
    • 删除数据
    - (void)deleteData{
       
        //创建删除请求
        NSFetchRequest *deleRequest = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
        
        //删除条件
        NSPredicate *pre = [NSPredicate predicateWithFormat:@"age < %d", 10];
        deleRequest.predicate = pre;
       
        //返回需要删除的对象数组
        NSArray *deleArray = [_context executeFetchRequest:deleRequest error:nil];
        
        //从数据库中删除
        for (Student *stu in deleArray) {
            [_context deleteObject:stu];
        }
       
        NSError *error = nil;
        //保存--记住保存
        if ([_context save:&error]) {
            [self alertViewWithMessage:@"删除 age < 10 的数据"];
        }else{
            NSLog(@"删除数据失败, %@", error);
        }
    }
    
    • 更新修改
    //更新,修改
    - (void)updateData{
        
        //创建查询请求
        NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
        NSPredicate *pre = [NSPredicate predicateWithFormat:@"sex = %@", @"帅哥"];
        request.predicate = pre;
        
        //发送请求
        NSArray *resArray = [_context executeFetchRequest:request error:nil];
        
        //修改
        for (Student *stu in resArray) {
            stu.name = @"且行且珍惜_iOS";
        }
      
        //保存
        NSError *error = nil;
        if ([_context save:&error]) {
            [self alertViewWithMessage:@"更新所有帅哥的的名字为“且行且珍惜_iOS”"];
        }else{
            NSLog(@"更新数据失败, %@", error);
        }  
    }
    
    • 读取查询
    //读取查询
    - (void)readData{
        
        /* 谓词的条件指令
         1.比较运算符 > 、< 、== 、>= 、<= 、!=
         例:@"number >= 99"
         
         2.范围运算符:IN 、BETWEEN
         例:@"number BETWEEN {1,5}"
         @"address IN {'shanghai','nanjing'}"
         
         3.字符串本身:SELF
         例:@"SELF == 'APPLE'"
         
         4.字符串相关:BEGINSWITH、ENDSWITH、CONTAINS
         例:  @"name CONTAIN[cd] 'ang'"  //包含某个字符串
         @"name BEGINSWITH[c] 'sh'"    //以某个字符串开头
         @"name ENDSWITH[d] 'ang'"    //以某个字符串结束
         
         5.通配符:LIKE
         例:@"name LIKE[cd] '*er*'"   //*代表通配符,Like也接受[cd].
         @"name LIKE[cd] '???er*'"
         
         *注*: 星号 "*" : 代表0个或多个字符
         问号 "?" : 代表一个字符
         
         6.正则表达式:MATCHES
         例:NSString *regex = @"^A.+e$"; //以A开头,e结尾
         @"name MATCHES %@",regex
         
         注:[c]*不区分大小写 , [d]不区分发音符号即没有重音符号, [cd]既不区分大小写,也不区分发音符号。
         
         7. 合计操作
         ANY,SOME:指定下列表达式中的任意元素。比如,ANY children.age < 18。
         ALL:指定下列表达式中的所有元素。比如,ALL children.age < 18。
         NONE:指定下列表达式中没有的元素。比如,NONE children.age < 18。它在逻辑上等于NOT (ANY ...)。
         IN:等于SQL的IN操作,左边的表达必须出现在右边指定的集合中。比如,name IN { 'Ben', 'Melissa', 'Nick' }。
         
         提示:
         1. 谓词中的匹配指令关键字通常使用大写字母
         2. 谓词中可以使用格式字符串
         3. 如果通过对象的key
         path指定匹配条件,需要使用%K
         
         */
        
        //创建查询请求
        NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
        //查询条件
        NSPredicate *pre = [NSPredicate predicateWithFormat:@"sex = %@", @"美女"];
        request.predicate = pre;
     
        // 从第几页开始显示
        // 通过这个属性实现分页
        //request.fetchOffset = 0;
        // 每页显示多少条数据
        //request.fetchLimit = 6;
    
        //发送查询请求
        NSArray *resArray = [_context executeFetchRequest:request error:nil];
        
        [self alertViewWithMessage:@"查询所有的美女"];  
    }
    
    
    • 排序
    //排序
    - (void)sort{
        //创建排序请求
        NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
        //实例化排序对象
        NSSortDescriptor *ageSort = [NSSortDescriptor sortDescriptorWithKey:@"age"ascending:YES];
        NSSortDescriptor *numberSort = [NSSortDescriptor sortDescriptorWithKey:@"number"ascending:YES];
        request.sortDescriptors = @[ageSort,numberSort];
        //发送请求
        NSError *error = nil;
        NSArray *resArray = [_context executeFetchRequest:request error:&error];
        if (error == nil) {
            [self alertViewWithMessage:@"按照age和number排序"];
        }else{
            NSLog(@"排序失败, %@", error);
        }
    }
    
    
    CoreData调试:

    打开Product,选择Edit Scheme.
    选择Arguments,在下面的ArgumentsPassed On Launch中添加下面两个选项,如图:
    (1)-com.apple.CoreData.SQLDebug
    (2)1

    CoreData调试.png

    参考文献:
    http://www.jianshu.com/p/880dd63c5f5e
    http://www.jianshu.com/p/79ba72810d29
    http://www.jianshu.com/p/3e4f33b5b013
    http://blog.csdn.net/u013263917/article/details/53277708
    http://blog.csdn.net/u013263917/article/details/53277863

    示例Github:CoreDataLearn

    更新于2018.3.7 iOS CoreData (二) 版本升级和数据库迁移

    相关文章

      网友评论

      • MrZombie:请问比如student.objectID是主键的意思吗?会不会变化
      • Kaaaaai:CoreData 是对 SQLite 的封装吗?应该不是吧
        Kaaaaai:@且行且珍惜_iOS 没什么高见,就是看了不同的文档,有的说底层是 SQLite,有的说不是,所以就问一下
        且行且珍惜_iOS:这位兄台有何高见呢?
      • 清风_____:异步处理会不会出现崩溃
      • Sxiaobai:NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:@"Student"];
        NSArray *resArray = [_context executeFetchRequest:request error:nil];
        _dataSource = [NSMutableArray arrayWithArray:resArray];
        [self.tableView reloadData];

        这个方法为啥要调那么多次呢:cry:
        游龙飞雪:NSArray *resArray = [_context executeFetchRequest:request error:nil]; 这是查询出列表。
      • long弟弟:楼主很认真的样子。看完一就想看二。iOS CoreData (一) 增删改查;iOS CoreData (二)数据库升级
        且行且珍惜_iOS:已经出二了,😁😀https://www.jianshu.com/p/38e6017ab1e9
      • 九林:Mark

      本文标题:iOS CoreData (一) 增删改查

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