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

时间:2017-09-17 11:16:38

标签: core-data frameworks swift4

我目前正在使用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

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

如何创建一个框架,在单个项目中的多个目标中创建共享Core Data数据库?

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

enter image description here

目标: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
    }

}

2 个答案:

答案 0 :(得分:2)

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

enter image description here

答案 1 :(得分:0)

我自己也碰到了这个,所以我想发布其他几个选项。

由于您没有对NSPersistentContainer进行子类化,因此它不知道在哪里查找包,因此除了查看主包之外别无选择。

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