我正在实现 Stanford CS193p 作业 (PhotoMania) 的 Swift 版本。
我很困惑为什么我的 NSObjectContext 保存命令没有保存到我认为我已经创建的持久存储中。
该应用程序的目的是有两个标签栏: - 标签栏 #1 包含来自 Flickr 的最新照片。 - 标签栏 #2 包含我最近查看的照片。这意味着每次用户点击 Flickr 照片单元格查看图像时,我应该将该图像的信息存储在持久存储结构中,以便在标签栏 #2 中访问它。
应用的结构如下:
1. AppDelegate 包含存储设置说明:
import UIKit
import CoreData
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
var objectContext:NSManagedObjectContext?
func application(application: UIApplication!, didFinishLaunchingWithOptions launchOptions: NSDictionary!) -> Bool {
self.objectContext = self.createMainQueueManagedObjectContext()
self.testModel()
return true
}
func testModel(){
var photoModel: Photo? = NSEntityDescription.insertNewObjectForEntityForName("Photo", inManagedObjectContext: self.objectContext!) as? Photo
photoModel!.title = "hi"
photoModel!.subtitle = "subtitle"
self.objectContext!.insertObject(photoModel)
self.objectContext!.save(nil)
println(self.objectContext!.persistentStoreCoordinator.persistentStores[0].objects)
}
func createManagedObjectModel()-> NSManagedObjectModel{
var managedObjectModel:NSManagedObjectModel?
let modelURL:NSURL = NSBundle.mainBundle().URLForResource("PhotoModel", withExtension: "momd")
managedObjectModel = NSManagedObjectModel(contentsOfURL: modelURL)
return managedObjectModel!
}
func createPersistentStoreCoordinator()-> NSPersistentStoreCoordinator{
var persistentStoreCoordinator:NSPersistentStoreCoordinator?
var managedObjectModel:NSManagedObjectModel = self.createManagedObjectModel()
let storeURL:NSURL = self.applicationDocumentsDirectory().URLByAppendingPathComponent("MOC.sqlite")
var error:NSError?
persistentStoreCoordinator = NSPersistentStoreCoordinator(managedObjectModel: managedObjectModel)
if (persistentStoreCoordinator?.addPersistentStoreWithType(NSSQLiteStoreType as String, configuration: nil, URL: storeURL, options: nil, error: &error) == nil){
NSLog("unresolved error %@, %@", error!, error!.userInfo)
abort()
}
return persistentStoreCoordinator!
}
func applicationDocumentsDirectory()-> NSURL{
var hello:NSURL = NSFileManager().URLsForDirectory(NSSearchPathDirectory.DocumentDirectory, inDomains: NSSearchPathDomainMask.UserDomainMask).last as NSURL
return hello
}
func createMainQueueManagedObjectContext()->NSManagedObjectContext{
var managedObjectContext:NSManagedObjectContext?
var coordinator:NSPersistentStoreCoordinator = self.createPersistentStoreCoordinator()
if (coordinator != nil){
managedObjectContext = NSManagedObjectContext(concurrencyType: NSManagedObjectContextConcurrencyType.MainQueueConcurrencyType)
managedObjectContext!.persistentStoreCoordinator = coordinator
}
return managedObjectContext!
}
}
2. 有一个 UITableView 子类,它从 AppDelegate 获取上下文:
class MyViewController: UITableViewController {
@IBOutlet weak var refresher: UIRefreshControl?
let appDelegate:AppDelegate = UIApplication.sharedApplication().delegate as AppDelegate
var objectContext:NSManagedObjectContext?{
didSet{
self.fetchDatabasePhotos()
}
}
var fetchController:NSFetchedResultsController?{
didSet{
self.tableView.reloadData()
}
}
override func viewDidLoad() {
self.objectContext = self.appDelegate.objectContext
super.viewDidLoad()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
func fetchDatabasePhotos(){
let request:NSFetchRequest = NSFetchRequest(entityName: "Photo")
request.predicate = nil
request.sortDescriptors = [NSSortDescriptor(key: "title", ascending: true)]
self.fetchController = NSFetchedResultsController(fetchRequest: request, managedObjectContext: self.objectContext, sectionNameKeyPath: nil, cacheName: nil)
println(self.fetchController!)
// println(self.fetchController!.fetchedObjects)
}
}
3. MyViewController 有一个“JustPostedFlickr”子类,用于管理 Flickr 下载。 Flickr 下载部分工作正常。但是我尝试通过 TableView Controller 和 ImageView Controller 之间的 prepareforsegue 中调用的方法在存储中插入照片对象。这是被调用的方法:
override func prepareForSegue(segue: UIStoryboardSegue!, sender: AnyObject!) {
let indexPath:NSIndexPath = self.tableView.indexPathForCell(sender as UITableViewCell)
if (segue.identifier == "Display Photo"){
if (segue.destinationViewController.isKindOfClass(ImageViewController)){
self.prepareImageViewController(segue.destinationViewController as ImageViewController, photo: self.photos[indexPath.row] as NSDictionary)
}
}
}
func prepareImageViewController(ivc:ImageViewController, photo:NSDictionary){
ivc.imageURL = FlickrFetcher.URLforPhoto(photo, format: FlickrPhotoFormatLarge)
ivc.title = photo.valueForKeyPath(FLICKR_PHOTO_TITLE) as String
var photoModel: Photo! = NSEntityDescription.insertNewObjectForEntityForName("Photo", inManagedObjectContext: self.objectContext) as Photo
photoModel!.title = photo.valueForKeyPath(FLICKR_PHOTO_TITLE) as String
photoModel!.subtitle = photo.valueForKeyPath(FLICKR_PHOTO_DESCRIPTION) as String
photoModel!.photoURL = FlickrFetcher.URLforPhoto(photo, format: FlickrPhotoFormatLarge).absoluteString
photoModel!.created_date = NSDate()
println(self.objectContext!)
var theVar:Bool = self.objectContext!.save(nil)
println(theVar)
self.fetchDatabasePhotos()
}
self.objectContext!.save(xx)
始终返回 true。
self.objectContext.registeredObjects
返回正确的照片模型结构和值
但是,当我获取照片并打印获取的对象列表时,它总是返回 nil。
标签栏 #2(最近的照片)的 Controller 应该从 fetchController 获取内容。 fetchcontroller.fetchedObjects
方法每次都返回 nil,即使在我点击图像(应该填充存储)之后也是如此:
import UIKit
class RecentPhotosController: MyViewController {
override func viewDidAppear(animated: Bool) {
self.objectContext = self.appDelegate.createMainQueueManagedObjectContext()
super.viewDidAppear(animated)
println("I just appeared")
}
override func numberOfSectionsInTableView(tableView: UITableView!) -> Int {
var sections:Int = 0
let fetchSections = self.fetchController!.sections
if (fetchSections != nil){
sections = fetchSections.count
}
return sections
}
override func tableView(tableView: UITableView!, numberOfRowsInSection section: Int) -> Int {
var rows:Int = 0
let fetchRows = self.fetchController!.fetchedObjects
if (fetchRows != nil){
rows = fetchRows.count
}
return rows
}
override func tableView(tableView: UITableView!, cellForRowAtIndexPath indexPath: NSIndexPath!) -> UITableViewCell! {
let cell = tableView.dequeueReusableCellWithIdentifier("Recent Photo", forIndexPath: indexPath) as UITableViewCell
let photo:Photo = self.fetchController!.objectAtIndexPath(indexPath)! as Photo
cell.textLabel.text = photo.title as String
cell.detailTextLabel.text = photo.subtitle as String
return cell
}
}
最后:这是照片模型:
import Foundation
import CoreData
class Photo: NSManagedObject {
@NSManaged var title: String
@NSManaged var subtitle: String
@NSManaged var created_date: NSDate
@NSManaged var photoURL: String
}
我的问题是:为什么 NSObjectContext 中的对象没有保存到存储中?
最后:我现在正在模拟器上进行测试,但我不希望这成为我的问题的根源,因为我实现了 CS193p 的 Photomania(使用核心数据和持久存储)的 Swift 版本并且运行良好。
最佳答案
您似乎有两个 Controller 类,每个类都调用 self.appDelegate.createMainQueueManagedObjectContext()
createMainQueueManagedObjectContext
为每个 Controller 创建一个新的上下文。这很好,但它还会调用 createPersistentStoreCoordinator
,它会为每个上下文创建一个新的持久存储协调器。这是一个问题,因为现在您有两个存储协调器,每个都有自己的持久存储,竞相写入磁盘上的同一个文件。谁知道会发生什么取决于每个商店何时读取或写入相对于另一个商店的文件。
您可能想要做的是:
- 使用由两个 Controller 共享的单个托管对象上下文。
- 创建一个持久存储协调器,将其提供给每个托管对象上下文,并为每个 Controller 提供自己的上下文。
关于尽管返回 True,IOS ObjectContext 仍未保存到持久存储,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25578407/