具有多个目标的核心数据框架

Core Data Framework With Multiple Targets

我目前正在 swift 4 中制作一个项目,它使用了一些目标。由于所有目标都需要访问相同的核心数据,所以我决定制作自己的框架目标,存储它的数据模型以及它的访问信息。

我遇到的问题是在我的应用程序目标 (CoreDataTest) 中,当我 运行 应用程序时,出现以下错误:

2017-09-17 12:02:20.787132+0100 CoreDataTest[22070:3218298] [error] error:  Failed to load model named TestData
CoreData: error:  Failed to load model named TestData
2017-09-17 12:02:20.787594+0100 CoreDataTest[22070:3218298] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'An NSManagedObject of class 'Message' must have a valid NSEntityDescription.'
*** First throw call stack:
(0x182097d38 0x1815ac528 0x18481a73c 0x1026c6678 0x1026c6708 0x1848bac5c 0x10261d480 0x10261ce94 0x10261cd48 0x10261cecc 0x18b42b96c 0x18b42b544 0x18b43210c 0x18b42f378 0x18b49edb4 0x18b68e570 0x18b693300 0x18b9129b4 0x18bbd90d0 0x18b912618 0x18b912e88 0x18c05daf4 0x18c05d998 0x18bde7ab4 0x18bf77c94 0x18bde7964 0x18bbd8730 0x18b691a44 0x18ba80144 0x18472d968 0x184736270 0x10344145c 0x10344db74 0x184761b04 0x1847617a8 0x184761d44 0x182040358 0x1820402d8 0x18203fb60 0x18203d738 0x181f5e2d8 0x183de3f84 0x18b48f5e0 0x10261deec 0x181a8256c)
libc++abi.dylib: terminating with uncaught exception of type NSException

我添加了一个异常断点,它在 PersistenceStorage.saveContext()ViewController.swift 中崩溃了。

我将如何创建一个框架来在单个项目的多个目标中创建共享的核心数据数据库?

这是我的项目设置。请注意,每个组都是自己的目标。

目标:CoreDataKit(框架)
CoreData.swift

import CoreData

public class PersistenceStorage {

    private init() {}

    public static var context: NSManagedObjectContext {
        return persistentContainer.viewContext
    }


    public static var persistentContainer: NSPersistentContainer = {
        let container = NSPersistentContainer(name: "TestData")
        container.loadPersistentStores(completionHandler: { (storeDescription, error) in
            if let error = error as NSError? {
                fatalError("Unresolved error \(error), \(error.userInfo)")
            }
        })
        return container
    }()


    public static func saveContext () {
        let context = persistentContainer.viewContext
        if context.hasChanges {
            do {
                try context.save()
            } catch {
                let nserror = error as NSError
                fatalError("Unresolved error \(nserror), \(nserror.userInfo)")
            }
        }
    }
}

留言+CoreDataClass.swift

import Foundation
import CoreData

@objc(Message)
public class Message: NSManagedObject {

}

留言+CoreDataProperties.swift

import Foundation
import CoreData


extension Message {

    @nonobjc public class func fetchRequest() -> NSFetchRequest<Message> {
        return NSFetchRequest<Message>(entityName: "Message")
    }

    @NSManaged public var text: String?

}

目标:CoreDataTest(主应用程序)
ViewController.swift

import UIKit
import CoreDataKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        let message = Message(context: PersistenceStorage.context)
        message.text = "test"
        PersistenceStorage.saveContext()

    }
}

AppDelegate.swift

import UIKit
import CoreData

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
        // Override point for customization after application launch.
        return true
    }

}

经过大量研究和反复试验,我发现我需要将 xcdatamodel 文件上的目标成员设置为我希望与之共享的其他目标。

我自己刚遇到这个问题,所以我想 post 还有其他几个选择。

由于您没有子类化 NSPersistentContainer,因此它不知道在哪里寻找包,所以别无选择,只能在主包中查找。

如果您只想在您的框架中拥有您的模型,那么您可以将 NSPersistentContainer 子类化,或者您可以先加载模型并将其传递给持久容器的初始化。如果您只有一个模型,那么执行类似 mergedModelFromBundles 的操作非常简单,然后将该模型传递给持久容器初始化程序。