core-data - 从多个模型构建 NSManagedObjectModel

标签 core-data swift2 nsmanagedobjectmodel

有人想要合并多个 NSManagedObjectModel 有几个原因。如果你在网上搜索,所有的回答都是不可能的,或者只有共享一种或多种关系的两个不相关的实体才有可能。请参阅thisthis例如链接。

但是,通过一点或更多的工作,(我认为)可以合并 NSManagedObjectModels,即使实体是相关的(如父子关系)或者属性分布在多个模型中。

尽管它不会在 Xcode 模型编辑器中轻松显示,并且开箱即用的转换(可能)不会起作用。

在下面的答案中,我对核心数据和合并多个模型的代码进行了观察。如果您发现任何错误或有改进建议,请在此处回复。

最佳答案

我注意到的一些事情:

  1. 复制 NSPropertyDescription(属性、关系)会复制其所有值,但不会复制其所属的实体。对于destinationEntity 和inverseRelationship 也是如此。

  2. 因此,应将复制的 NSPropertyDescription 添加到实体中。因此,该实体的所有子实体也会自动获取该属性。

  3. 复制 NSEntityDescription 不包括父实体。因此(NSManagedObjectEntity 的)树必须手动重建。

  4. 如果您设置实体的父级,则该(子)实体将立即自动继承其所有父级属性。 换句话说,当您向一个实体询问其属性时,该实体已经知道其所有属性。它不会首先查询其父级。 (合理假设)

  5. 向模型添加实体会填写目标实体以及所添加实体的关系描述的逆关系描述。

  6. 如果在使用任何实体或属性之前没有设置其名称,核心数据将会提示。 那就是copy by name而不是值(value)方面。

  7. 向已具有同名属性(来自自身或从其祖先继承)的实体添加属性将使核心数据产生提示。

这会转化为以下代码:

extension NSPropertyDescription
{
   var isPlaceholder : Bool { return self.userInfo?["isPlaceholder"] != nil }
}

extension NSEntityDescription
{
   var isPlaceholder : Bool { return self.userInfo?["isPlaceholder"] != nil }
}

func mergeModels(models: [NSManagedObjectModel]) -> NSManagedObjectModel?
{
    var entities : [String : NSEntityDescription] = [:]

    //support functions
    let makeEntity : String -> NSEntityDescription = { entityName in
        let newEntity = NSEntityDescription()
        entities[entityName] = newEntity
        newEntity.name = entityName
        return newEntity
    }

    let setParent : (String, NSEntityDescription) -> () = { parentName, child in
        if let parent = entities[parentName]
        {
            parent.subentities.append(child)
        }
        else //parent has not yet been encountered, so generate it
        {
            let newParentEntity = makeEntity(parentName)
            newParentEntity.subentities.append(child)
        }
    }


    //rebuild model: generate new description for each entity and add non-placeholder properties
    for model in models
    {
        for entity in model.entities
        {
            guard let entityName = entity.name else { fatalError() }
            let mergedEntity = entities[entityName] ?? makeEntity(entityName)

            //set entity properties
            if !entity.isPlaceholder
            {
                mergedEntity.abstract = entity.abstract
                mergedEntity.managedObjectClassName = entity.managedObjectClassName
            }

            //set parent, if any
            if mergedEntity.superentity == nil, //no parent set
                let parentName = entity.superentity?.name //but parent is required
            {
                setParent(parentName, mergedEntity)
            }

            //set properties
            for property in entity.properties
            {
                if property.isPlaceholder ||
                    mergedEntity.properties.contains({$0.name == property.name})
                { continue }

                let newProperty = property.copy() as! NSPropertyDescription
                mergedEntity.properties.append(newProperty)

            }
        }
    }

    //generate final model
    let mergedModel = NSManagedObjectModel()
    mergedModel.entities = Array(entities.values) //sets the destination entity and inverse relationship descriptions
    return mergedModel
}

在 ManagedObjectModel(xcode 编辑器)中,在实体和/或属性的用户信息字典中设置“占位符”标志。

可以通过在用户信息字典中设置附加键来指定哪个模型具有主要实体/属性/关系(设置)并适当调整此代码片段来细化模型生成。

但是,如果您可以避免使用多个模型,那就避免它。通过坚持标准的单一模型方法,您的生活将会变得更加简单。

[免责声明:据我所知,这段代码应该可以工作。但不能保证。]

关于core-data - 从多个模型构建 NSManagedObjectModel,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34250864/

相关文章:

Swift Core 数据按天获取结果并排序到带有时间部分的表格 View 中

ios - 在swift 2.2中按第二个字母对数组进行排序

ios - 核心数据保存模型不适用于 iOS

ios - 存储元数据和托管对象模型之间的兼容性标准是什么?

ios - NSSortDescriptor 从 Swift 4 的核心数据中获取键

objective-c - UIManagedDocument 上的 saveToURL 核心数据失败

ios - 如何将框架中的 UIView 子类设置为 Storyboard自定义类?

swift - 在 Xcode 8 中使用 Swift 2.2?

ios - 在CoreData中存储 "Item-ItemDetails"的最佳方式

iphone - 在 UITableView 中创建运行总和