NSKeyedUnArchiver崩溃

时间:2016-11-29 14:40:48

标签: ios swift appdelegate nsexception nskeyedunarchiver

由于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")
    }

}

1 个答案:

答案 0 :(得分:-1)

你没有显示你的Photomania.Item类/结构但它应该符合NSCoding协议。看看这个例子: http://mhorga.org/2015/08/25/ios-persistence-with-nscoder-and-nskeyedarchiver.html