尽管返回 True,IOS ObjectContext 仍未保存到持久存储

标签 ios uitableview core-data swift nsmanagedobjectcontext

我正在实现 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,它会为每个上下文创建一个新的持久存储协调器。这是一个问题,因为现在您有两个存储协调器,每个都有自己的持久存储,竞相写入磁盘上的同一个文件。谁知道会发生什么取决于每个商店何时读取或写入相对于另一个商店的文件。

您可能想要做的是:

  1. 使用由两个 Controller 共享的单个托管对象上下文。
  2. 创建一个持久存储协调器,将其提供给每个托管对象上下文,并为每个 Controller 提供自己的上下文。

关于尽管返回 True,IOS ObjectContext 仍未保存到持久存储,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25578407/

相关文章:

ios - 如何在 iOS Gmail 应用程序中禁用邮件修改

iphone - 在 XCode 4.2 空应用程序模板中使用 Interface Builder 将 tabbarcontroller 添加到 AppDelegate 的步骤

swift - TableViewController 仅显示一个动态部分

objective-c - 如何将获取的对象保存到核心数据

swift - 如何使用 SwiftUI 和 CoreData (@FetchRequest) 对动态列表排序进行动画处理

ios - 具有圆角半径的 UILabel View 在 iOS 17 Beta 上不起作用 - Xcode 15 Beta

iphone - 滚动图像的巨大延迟

ios - UITableview 下的滑动工具栏

ios - 带有图像的 UITableview 滚动速度非常慢

objective-c - NSManagedObject 的 willSave 未被无效对象调用