2017-09-17 2 views
0

Je réalise actuellement un projet dans swift 4 qui utilise quelques cibles. Étant donné que toutes les cibles doivent accéder aux mêmes données de base, j'ai décidé de créer ma propre cible de structure qui stocke le modèle de données ainsi que les informations d'accès correspondantes.Cadre de données de base avec plusieurs cibles

Le problème que j'ai est dans mon objectif d'application (CoreDataTest), quand je lance l'application, je reçois l'erreur suivante:

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 0x18bc 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 

J'ai ajouté un point de rupture d'exception et il se bloque dans ViewController.swift à PersistenceStorage.saveContext() . Comment créer un cadre pour créer une base de données de base de données partagée dans mes multiples cibles dans un même projet?

Voici ma configuration de projet. Veuillez noter que chaque groupe est sa propre cible.

enter image description here

cible: CoreDataKit (cadre)
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)") 
      } 
     } 
    } 
} 

message + CoreDataClass.swift

import Foundation 
import CoreData 

@objc(Message) 
public class Message: NSManagedObject { 

} 

message + 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? 

} 

Cible: CoreDataTest (Application principale)
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 
    } 

} 

Répondre

1

Après beaucoup de recherches et de tâtonnements, je trouve que je dois régler la cible les membres du fichier xcdatamodel vers les autres cibles avec lesquelles je voulais le partager.

enter image description here

0

Juste cogné dans moi-même donc pensé que je poste un ou deux autres options. Puisque vous ne sous-classez pas NSPersistentContainer, alors il ne sait pas où chercher les paquets, donc il n'a pas d'autre choix que de regarder dans le paquet principal. Si vous souhaitez uniquement avoir votre modèle dans votre framework, vous pouvez sous-classer NSPersistentContainer, ou vous pouvez charger le modèle en premier et le passer à l'initialisation du conteneur persistant. Si vous n'avez qu'un seul modèle, alors faire quelque chose comme mergedModelFromBundles est assez simple à faire et ensuite passer ce modèle à l'initialiseur de conteneur persistant.