CoreData 基础

  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

CoreData 基础_第1张图片

    为Person实体生成modal类

CoreData 基础_第2张图片


3.    创建相关VC

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

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

CoreData 基础_第3张图片

    效果图大致如下:

CoreData 基础_第4张图片      CoreData 基础_第5张图片


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 的差异及性能对比。

 

 

你可能感兴趣的:(ios,coredata)