文档章节

CoreData 基础

Carson6931
 Carson6931
发布于 2015/06/14 19:21
字数 1167
阅读 181
收藏 3
  1.  新建工程勾选 "Use CoreData" 你会发现AppDelegate  文件多了一些属性及方法:

- (NSManagedObjectContext*)managedObjectContext

        主要用于获取 “被管理的上下文”,将此方法放在AppDelegate 的原因也是显而易见的,那就是需要保持统 一, 既然使用CoreData 那就注意始终使用同一个被管理的上下文,简单的就是同一个对象。

        NSManagedObjectContext:被管理的上下文

       作用:操作上下文(增删改查)

- (NSPersistentStoreCoordinator *)persistentStoreCoordinator

         NSPersistentStoreCoordinator : 持久化存储助理

          作用:相当于数据库的连接器

- (NSManagedObjectModel *)managedObjectModel

        NSManagedObjectModel: 被管理的数据模型

         作用:管理实体、维护实体间的关系

通过代码的执行顺序我们发现,通过NSManagedObjectModel 得到 NSPersistentStoreCoordinator 通过 NSPersistentStoreCoordinator 得到 NSManagedObjectContext

NSManagedObjectModel ——>NSPersistentStoreCoordinator——>NSManagedObjectContext

2.   新建Person 实体 同时创建属性name surName

    为Person实体生成modal类


3.    创建相关VC

    这里就简单介绍关于Person 实体的 新增、修改、删除,所以这里需要再建两个类:PersonTVC(person 实体展示) PeronEditTVC(新增、编辑person)

那UI也很简单,storyboard快速创建相关视图即可,为了方便person 及personEdit是UITableViewController 子类

    效果图大致如下:

     


4、实现相关代码

  • PersonTVC.h:

#import <UIKit/UIKit.h>
#import <CoreData/CoreData.h>
@interface PersonTVC : UITableViewController<NSFetchedResultsControllerDelegate>
//获取结果控制器
@property (strong ,nonatomic) NSFetchedResultsController *fetchedResultController;
//上下文对象
@property (strong, nonatomic) NSManagedObjectContext *managedObjectContext;
@property (nonatomic, assign) BOOL beganUpdates;
@end

    NSFetchedResultsController 为系统封装好的 “获取结果控制器”,主要是配合table使用方便,当然你也可以自己去fetch,拿到数组,显示。

    NSManagedObjectContext 被管理的上下文,应与Appdelegate里面的上下文对象一致,如何获取就不多说了...

  • PersonTVC.m

#pragma mark - 初始化fetchResultController
- (void)setupFetchedResultsController
{
//    NSBatchUpdateReuqest
    // 1 - Decide what Entity you want
    NSString *entityName = @"Person"; // Put your entity name here
    
    // 2 - Request that Entity
    NSFetchRequest *request = [NSFetchRequest fetchRequestWithEntityName:entityName];
    request.sortDescriptors = [NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:@"name"
                                                                                     ascending:YES
                                                                                      selector:@selector(localizedCaseInsensitiveCompare:)]];
    _fetchedResultController = [[NSFetchedResultsController alloc]initWithFetchRequest:request managedObjectContext:_managedObjectContext sectionNameKeyPath:nil cacheName:nil];
    _fetchedResultController.delegate = self;
    // 5 - Fetch it
    [self performFetch];
}

    得到 “获取结果控制器对象”,并将结果按Name 字母排序,也可以不排#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
    // Return the number of sections.
    return [[_fetchedResultController sections]count];
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    // Return the number of rows in the section.
    return [[[_fetchedResultController sections]objectAtIndex:section]numberOfObjects];
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"personCell" forIndexPath:indexPath];
    
    Person *person = [self.fetchedResultController objectAtIndexPath:indexPath];
    NSString *allName = [NSString stringWithFormat:@"%@ %@",person.name,person.surName];
    cell.textLabel.text = allName;
    return cell;
}
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath {
    
    if (editingStyle == UITableViewCellEditingStyleDelete) {
        
        [self.tableView beginUpdates]; // Avoid  NSInternalInconsistencyException
        
        // Delete the person object that was swiped
        Person *personToDelete = [_fetchedResultController objectAtIndexPath:indexPath];
        [self.managedObjectContext deleteObject:personToDelete];
        [self.managedObjectContext save:nil];
        
        // Delete the (now empty) row on the table
        [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
        [self performFetch];
        
        [self.tableView endUpdates];
    }
}

    Table 的代理方法,基本上可以看出,数据源是从 _fetchedResultController 中获取。使用_fetchedResultController的好处还有 它实现了自动数据的自动监听回调,我们能够及时根据数据变化来刷新table。

  • _fetchedResultController 的代理方法实现

#pragma mark - NSFetchedResultsControllerDelegate
- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller
{
    [self.tableView beginUpdates];
    _beganUpdates = YES;
}
- (void)controller:(NSFetchedResultsController *)controller
  didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo
           atIndex:(NSUInteger)sectionIndex
     forChangeType:(NSFetchedResultsChangeType)type
{
    switch(type)
    {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;
            
        case NSFetchedResultsChangeDelete:
            [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
            break;
        default:
            break;
    }
}
- (void)controller:(NSFetchedResultsController *)controller
   didChangeObject:(id)anObject
       atIndexPath:(NSIndexPath *)indexPath
     forChangeType:(NSFetchedResultsChangeType)type
      newIndexPath:(NSIndexPath *)newIndexPath
{
    switch(type)
    {
        case NSFetchedResultsChangeInsert:
            [self.tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
            
        case NSFetchedResultsChangeDelete:
            [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
            
        case NSFetchedResultsChangeUpdate:
            [self.tableView reloadRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
            
        case NSFetchedResultsChangeMove:
            [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
            [self.tableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
            break;
    }
}
- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller
{
    if (_beganUpdates) [self.tableView endUpdates];
}

    可以看出这些代理方法能够很好的监听每个状态的变化,这样我们不必自己去实现监听。

  • PersonEditTVC

    该类主要用于新增,和修改person 对象,所以定义好type 来区分,该类的重点在save 方法的实现

- (IBAction)saveEditPersonClick:(id)sender {
    if(_personType == personTypeAdd)
    {
        Person *person = [NSEntityDescription insertNewObjectForEntityForName:@"Person" inManagedObjectContext:_managedObjectContext];
        if(_nameField.text.length>0&&_surNameField.text.length>0)
        {
            person.name = _nameField.text;
            person.surName = _surNameField.text;
        }
    }
    else if(_personType == personTypeEidt)
    {
        _currentPerson.name = _nameField.text;
        _currentPerson.surName = _surNameField.text;
    }
    
    [_managedObjectContext save:nil];
    [self.navigationController popViewControllerAnimated:YES];
}

可以看到新增或修改的对象都要是当前上下文_managedObjectContext 的对象,修改或添加后只需调用_managedObjectContext save 即可。当上下文数据有变化时在PersonTVC 中 NSFetctchedResultsControllerDelegete 方法就会执行。实现table 实时刷新。

同样这里实现了新增,修改,同样还有删除。为了方便,直接在PersonTVC 上实现左滑删除。

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath {
    
    if (editingStyle == UITableViewCellEditingStyleDelete) {
        
        [self.tableView beginUpdates]; // Avoid  NSInternalInconsistencyException
        
        // Delete the person object that was swiped
        Person *personToDelete = [_fetchedResultController objectAtIndexPath:indexPath];
        [self.managedObjectContext deleteObject:personToDelete];
        [self.managedObjectContext save:nil];
        
        // Delete the (now empty) row on the table
        [self.tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
        [self performFetch];
        
        [self.tableView endUpdates];
    }
}

可以看出,同样也是拿到_fetchedResutController 中的具体对象,然后在上下文_managedObjectContext 中删除。

——————————————————————————

这里简单介绍了CoreData的基本使用方法,有人说用sqlite+FMDB 效果好,也有人支持用CoreData,下篇介绍coreData 与 sqlite 的差异及性能对比。

 

 

© 著作权归作者所有

共有 人打赏支持
Carson6931
粉丝 17
博文 9
码字总数 7471
作品 0
徐汇
程序员
私信 提问
CoreData基本使用(更新:储存数组、字典类型)

前几天项目里面加入了CoreData,第一次用,记录一下,这里只记录创建项目之后添加的CoreData。 demo地址: CoreDataDemo 1. 创建CoreData模型文件 2. 创建项目中需要用到的实体(Entities),...

YvanLiu
05/02
0
0
Swift实践:使用CoreData完成上班签到小工具

image.png 之前在前两篇里面实现了一个十分简陋的通讯录,而且都是通过系统默认的方式创建的CoreData。可是实际中哪里有那么好的事情嘛,要是忘记在创建工程的时候勾选了下面这个图怎么办? ...

非典型技术宅
2017/12/03
0
0
Core Data 基本应用

操作coredata 处理4个场景的情况 Core Data 调试模式 简单的实际操作 建立CoreData可视化管理文件,生成对应的实体类 在appDelegate中建立CoreData对应的管理类对象 Document的路径 建立NSM...

云飞扬v5
2016/09/21
0
0
使用CoreData的轻量级自动数据迁移

使用IOS的CoreData进行存储很方便,CoreData已经帮我们做了很多基础的工作,一般情况下没必要自己操作数据库了。 在开发中修改了实体模型,可能会遇到schema incompatibility的错误,这是因为...

Megan_zhou
2013/07/24
0
0
Core Data 学习笔记(一)框架简介

目录 Core Data 学习笔记(一)框架简介 Core Data 学习笔记(二)被管理对象模型 Core Data 学习笔记(三)被管理对象 Core Data 学习笔记 应用示例 Core Data 学习笔记 常用链接整理 零、前...

灰大羊
2015/07/26
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Windows / Linux / MacOS 设置代理上网的方法汇总

本文汇总了 Windows / Linux / MacOS 设置代理上网的各种方法,总结如下: 1、设置系统代理(Windows、Linux、MacOS) 2、设置代理插件(Chrome、Chromium、Firefox、Opera、QQ等浏览器) 3、...

sunboy2050
昨天
1
0
自定义 Maven 的 repositories

有时,应用中需要一些比较新的依赖,而这些依赖并没有正式发布,还是处于milestone或者是snapshot阶段,并不能从中央仓库或者镜像站上下载到。此时,就需要 自定义Maven的<repositories>。 ...

waylau
昨天
1
0
徒手写一个es6代码库

mkdir democd demonpm initnpm install -g babelnpm install -g babel-clinpm install --save-dev babel-preset-es2015-node5 在项目目录创建两个文件夹 functional-playground ......

lilugirl
昨天
2
0
linux定位应用问题的一些常用命令,特别针对内存和线程分析的dump命令

1.jps找出进程号,找到对应的进程号后面才好继续操作 2.linux查看进程详细信息 ps -ef | grep 进程ID 3. dump内存信息 Jmap -dump:format=b,file=YYMMddhhmm.dump pid 4.top查看cpu占用信息 ...

noob_chr
昨天
2
0
Android TV开发-按键焦点

写在前面 按键焦点过程了解 2.1 dispatchKeyEvent 过程了解 2.2 焦点查找请求过程了解 1.2.1 第一次获取焦点 1.2.3 按键焦点 焦点控制 焦点记忆 应用场景 参考资料 [TOC] 1. 写在前面 工...

冰雪情缘l
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部