【问题标题】:Building a NSManagedObjectModel from several models从多个模型构建 NSManagedObjectModel
【发布时间】:2015-12-13 12:02:54
【问题描述】:

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

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

虽然它不会在 Xcode 模型编辑器中轻易显示,而且开箱即用的转换(可能)不起作用。

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

【问题讨论】:

    标签: core-data swift2 nsmanagedobjectmodel


    【解决方案1】:

    我注意到的一些事情:

    1. 复制 NSPropertyDescription(属性、关系)会复制它的所有值,但不会复制它所属的实体。 destinationEntity 和 inverseRelationship 相同。

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

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

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

    5. 将实体添加到模型中会填写目标实体和添加实体的关系描述的反向关系描述。

    6. 如果在使用之前没有设置任何实体或属性的名称,核心数据会报错。 那是copy by name,而不是价值方面。

    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 编辑器)中,在实体和/或属性的用户信息字典中设置“占位符”标志。

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

    但是,如果您可以避免使用多个模型,请避免使用它。坚持标准的单一模型方法,您的生活会简单得多。

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

    【讨论】:

      【解决方案2】:

      NSManagedObjectModel 类具有以下工厂方法/构造函数

      class func mergedModel(from: [Bundle]?)
      class func mergedModel(from: [Bundle]?, forStoreMetadata: [String : Any])
      init?(byMerging: [NSManagedObjectModel]?)
      init?(byMerging: [NSManagedObjectModel], forStoreMetadata: [String : Any])
      

      可选的forStoreMetadata 属性允许指定模型的版本。

      https://developer.apple.com/documentation/coredata/nsmanagedobjectmodel

      我怀疑这些方法在操作人员询问和回答问题时不可用。)

      【讨论】:

      • 这里我的记忆很模糊,但是要注意模型是如何合并的。我将一个大型模型分解为多个文件,然后根据用户信息字典中定义的父级和关系占位符加入它们。框架函数不这样做。如果一个模型中的实体与另一个模型中的实体完全解耦,则这些工作正常。您应该使用什么取决于您的型号。
      猜你喜欢
      • 2013-06-21
      • 2017-12-09
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2010-12-09
      • 2010-09-12
      • 2023-04-06
      • 1970-01-01
      相关资源
      最近更新 更多