iOS CoreData详解(6)深入理解数据模型

iOS CoreData详解(六)深入理解数据模型

原创Blog,转载请注明出处
blog.csdn.net/hello_hwc
之前的五篇文章
1. 堆栈与基本概念
2. 基本操作
3. 与Tableview协作-NSFetchedResultController
4. CoreData原理Faulting and Uniquing
5. CoreData与多线程


前言:Coredata是一个良好的对象图管理工具。那么对象图也就是实体(Entity)的理解就尤为重要。实体包括两个部分,属性和实体之间的关系.
这是本文要讲解的数据模型
iOS CoreData详解(6)深入理解数据模型

  • 一个员工隶属于一个部门
  • 一个部门有多个员工

属性(Attributes)

一个属性的截图
iOS CoreData详解(6)深入理解数据模型
其中
属性分为很多种

  1. Optional 可选/必须

    很好理解,就是可选属性可以不设置,但是必须的属性就必须设置,否则在存储的时候会失败。这里要注意一点,可选属性不要设置默认值,因为会引起混淆。

  2. Transient 瞬态

    这个属性是很有用的,它的意义是除了不持久化到本地外,其他的与完全参与到对象图的管理中。这比临时变量好多了,因为支持undo,支持对象图管理

  3. Indexed索引

    和数据库的索引类似,索引能够大幅度提高查询速度,但是会增大表的大小,也会降低写入的速度。因为每次写入都要相应的更新索引

  4. Validation 验证

    可以进行一些简单的验证,例如图中对于字符串可以验证长度。以及String是否符合一个正则表达式。

更复杂的验证要重写KVC代理的方法(验证的例子来自于文档)
属性层次的验证,重写
-(BOOL)validate<Key>:(id *)ioValue error:(NSError **)outError
例如验证一个Age属性

-(BOOL)validateAge:(id *)ioValue error:(NSError **)outError {

    if (*ioValue == nil) {
        // trap this in setNilValueForKey? new NSNumber with value 0?
        return YES;
    }
    if ([*ioValue floatValue] <= 0.0) {
        if (outError != NULL) {
            NSString *errorStr = NSLocalizedStringFromTable(
                @"Age must greater than zero", @"Employee",
                @"validation: zero age error");
            NSDictionary *userInfoDict = @{ NSLocalizedDescriptionKey : errorStr };
            NSError *error = [[NSError alloc] initWithDomain:EMPLOYEE_ERROR_DOMAIN
                code:PERSON_INVALID_AGE_CODE
                userInfo:userInfoDict];
            *outError = error;
        }
        return NO;
    }
    else {
        return YES;
    }

当然,也支持其他验证,例如插入删除更新的时候验证。
这种验证的顺序首先要验证super,然后验证自己的逻辑,如果二者都有错误,则要把两个错误合并然后返回。例如

- (BOOL)validateForInsert:(NSError **)error
{
    BOOL propertiesValid = [super validateForInsert:error];
    // could stop here if invalid
    BOOL consistencyValid = [self validateConsistency:error];
    return (propertiesValid && consistencyValid);
}

- (BOOL)validateForUpdate:(NSError **)error
{
    BOOL propertiesValid = [super validateForUpdate:error];
    // could stop here if invalid
    BOOL consistencyValid = [self validateConsistency:error];
    return (propertiesValid && consistencyValid);
}


- (BOOL)validateConsistency:(NSError **)error
{
    static     NSCalendar *gregorianCalendar;

    BOOL valid = YES;
    NSDate *myBirthday = [self birthday];

    if ((myBirthday != nil) && ([[self hasDrivingLicense] boolValue] == YES)) {

        if (gregorianCalendar == nil) {
            gregorianCalendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
        }
        NSDateComponents *components = [gregorianCalendar components:NSYearCalendarUnit
                                                            fromDate:myBirthday
                                                              toDate:[NSDate date]
                                                             options:0];
        int years = [components year];

        if (years < 16) {

            valid = NO;

            // don't create an error if none was requested
            if (error != NULL) {

                NSBundle *myBundle = [NSBundle bundleForClass:[self class]];
                NSString *drivingAgeErrorString = [myBundle localizedStringForKey:@"TooYoungToDriveError"
                                  value:@"Person is too young to have a driving license."
                                  table:@"PersonErrorStrings"];

                NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
                [userInfo setObject:drivingAgeErrorString forKey:NSLocalizedFailureReasonErrorKey];
                [userInfo setObject:self forKey:NSValidationObjectErrorKey];

                NSError *drivingAgeError = [NSError errorWithDomain:PERSON_DOMAIN
                                                               code:NSManagedObjectValidationError
                                                           userInfo:userInfo];

                // if there was no previous error, return the new error
                if (*error == nil) {
                    *error = drivingAgeError;
                }
                // if there was a previous error, combine it with the existing one
                else {
                    *error = [self errorFromOriginalError:*error error:drivingAgeError];
                }
            }
        }
    }
    return valid;

如何合并两个NSError,返回一个NSError

  • 含有多个NSError的error code是code:NSValidationMultipleErrorsError
  • 把多个errors添加到一个数组,然后设置NSDetailedErrorsKey,这个key就是可以对应获得的error数组。
- (NSError *)errorFromOriginalError:(NSError *)originalError error:(NSError *)secondError
{
    NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
    NSMutableArray *errors = [NSMutableArray arrayWithObject:secondError];

    if ([originalError code] == NSValidationMultipleErrorsError) {

        [userInfo addEntriesFromDictionary:[originalError userInfo]];
        [errors addObjectsFromArray:[userInfo objectForKey:NSDetailedErrorsKey]];
    }
    else {
        [errors addObject:originalError];
    }

    [userInfo setObject:errors forKey:NSDetailedErrorsKey];

    return [NSError errorWithDomain:NSCocoaErrorDomain
                               code:NSValidationMultipleErrorsError
                           userInfo:userInfo];
}

实体之间的关系(Relationship)

iOS CoreData详解(6)深入理解数据模型
关系分为三种

  1. one to one 一对一;例如
  2. one to many 一对多
  3. many to many 多对多

关系还有一个属性是Inverse,除了特殊情况,是一定要选的,这个属性表示了对应关系。也是CoreData进行对象图维护的依据。


关系的删除规则-Delete Rule

Deny

关系的destination中只要有一个对象,就不能删除,例如如果还有一个员工,就不能删除部门

Nullify

删除源头后,destination对应的都设为nil(只在逆向关系Optional的时候有效)。例如,删除一个部门,则把部门中的员工对应的部门信息都设为nil

Cascade

删除源头后,删除destination所有对象。删除部门了以后,删除所有的员工对象。

NoAction

删除源头后,对Destination不做任何操作
在Destination中有大量对象的时候有用。
这种情况很少用,因为要自己维护对象图