由于NSException崩溃,我的app委托正在崩溃。我使用异常断点工具在NSUnarchiver期间以及名为ItemStore的文件中查明崩溃的位置(在init(()内部添加注释)。请帮我解决这个问题,我已经尝试了所有的想法。
没有异常断点,这就是崩溃
2016-11-29 09:46:42.546 Foodie[3100:386241] *** Terminating app due to uncaught exception 'NSInvalidUnarchiveOperationException', reason: '*** -[NSKeyedUnarchiver decodeObjectForKey:]: cannot decode object of class (Photomania.Item) for key (NS.objects); the class may be defined in source code or a library that is not linked'
当我使用异常断点时,我在ItemStore下面的已加星标的行中收到异常。
ItemStore.swift:
import Foundation
class ItemStore {
var allItems: [Item] = []
let itemArchiveURL: NSURL = {
let documentsDirectories =
NSFileManager.defaultManager().URLsForDirectory(.DocumentDirectory,
inDomains: .UserDomainMask)
let documentDirectory = documentsDirectories.first!
return documentDirectory.URLByAppendingPathComponent("items.archive")
}()
init() {
// here is the crash
if let archivedItems =
NSKeyedUnarchiver.unarchiveObjectWithFile(itemArchiveURL.path!) as? [Item] {
allItems += archivedItems
}
}
func moveItemAtIndex(fromIndex: Int, toIndex: Int) {
if fromIndex == toIndex {
return
}
// Get reference to object being moved so you can re-insert it
let movedItem = allItems[fromIndex]
// Remove item from array
allItems.removeAtIndex(fromIndex)
// Insert item in array at new location
allItems.insert(movedItem, atIndex: toIndex)
}
func createItem() -> Item {
let newItem = Item(random: true)
allItems.append(newItem)
return newItem
}
func removeItem(item: Item) {
if let index = allItems.indexOf(item) {
allItems.removeAtIndex(index)
}
}
func saveChanges() -> Bool {
print("Saving items to: \(itemArchiveURL.path!)")
return NSKeyedArchiver.archiveRootObject(allItems, toFile: itemArchiveURL.path!)
}
}
App Delegate:
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
// Override point for customization after application launch.
//Create an ImageStore
//let imageStore = ImageStore()
//let itemStore = ItemStore()
//Access the ItemsViewController and set its item store
//let navController = window!.rootViewController as! UINavigationController
//let itemsController = navController.topViewController as! ItemsViewController
//itemsController.itemStore = itemStore
//itemsController.imageStore = imageStore
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
let rootController = window?.rootViewController
if rootController is UITabBarController {
let firstTabItem = (rootController as! UITabBarController).viewControllers?[0]
if firstTabItem is UINavigationController {
let firstController = (firstTabItem as! UINavigationController).viewControllers.first as! ItemsViewController
firstController.itemStore = ItemStore()
firstController.imageStore = ImageStore()
}
}
return true
}
...
}
Photomania.Item:
import UIKit
class Item: NSObject, NSCoding {
var meal: String
var restaurantName: String?
var valueInDollars: Int
let dateCreated: NSDate
let itemKey: String
init(meal: String, restaurantName: String?, valueInDollars: Int) {
self.meal = meal
self.restaurantName = restaurantName
self.valueInDollars = valueInDollars
self.dateCreated = NSDate()
self.itemKey = NSUUID().UUIDString
}
convenience init(random: Bool = false) {
if random {
let nouns = ["Meal"]
let places = ["Restaurant"]
var idx = arc4random_uniform(UInt32(nouns.count))
let randomNoun = nouns[Int(idx)]
idx = arc4random_uniform(UInt32(places.count))
let randomPlace = places[Int(idx)]
let randomName = "\(randomNoun)"
let randomValue = Int(arc4random_uniform(100))
let randomRestaurantName = "\(randomPlace)"
self.init(meal: randomName,
restaurantName: randomRestaurantName,
valueInDollars: randomValue)
}
else {
self.init(meal: "", restaurantName: nil, valueInDollars: 0)
}
}
required init(coder aDecoder: NSCoder) {
meal = aDecoder.decodeObjectForKey("meal") as! String
dateCreated = aDecoder.decodeObjectForKey("dateCreated") as! NSDate
itemKey = aDecoder.decodeObjectForKey("itemKey") as! String
restaurantName = aDecoder.decodeObjectForKey("restaurantName") as! String?
valueInDollars = aDecoder.decodeIntegerForKey("valueInDollars")
super.init()
}
func encodeWithCoder(aCoder: NSCoder) {
aCoder.encodeObject(meal, forKey: "meal")
aCoder.encodeObject(dateCreated, forKey: "dateCreated")
aCoder.encodeObject(itemKey, forKey: "itemKey")
aCoder.encodeObject(restaurantName, forKey: "restaurantName")
aCoder.encodeInteger(valueInDollars, forKey: "valueInDollars")
}
}
答案 0 :(得分:-1)
你没有显示你的Photomania.Item类/结构但它应该符合NSCoding协议。看看这个例子: http://mhorga.org/2015/08/25/ios-persistence-with-nscoder-and-nskeyedarchiver.html