Попытка заархивировать экземпляр класса, соответствующего NSCoder

Я работаю над своим первым приложением Swift для iOS, у меня возникают проблемы с сериализацией и сохранением объекта, JSON которого я получаю с сервера. Я использую Gloss, облегченную библиотеку анализа JSON, которая определяет декодируемый протокол, с помощью которого можно создать экземпляр экземпляра. из JSON. Мое намерение состоит в том, чтобы загрузить вещь из JSON (псевдоним типа для [String : AnyObject]), сначала извлекая ее идентификатор, а затем проверяя, есть ли у меня уже локальная архивная копия. Если я это сделаю, разархивируйте это и получите изображение. Если нет, сделайте асинхронный запрос файла изображения.

Проблема в том, что Thing.localArchiveExists(id) всегда возвращает false. Вещи успешно создаются, но они всегда повторно извлекают изображение. Я проверил файловую систему и подтвердил, что архивные файлы не записываются. Однако я не вижу «ОШИБКА. Не удалось заархивировать», что говорит мне о том, что сохранение прошло успешно. Я что-то упустил о том, как архивировать и сохранять объекты NSCoder? Спасибо!

Вот моя реализация протокола Decodable:

// MARK: Decodable protocol
// When a thing is loaded from JSON, we load its image from archive if possible.
required init?(json: JSON) {
    guard let id: Int = "id" <~~ json else { return nil}

    if Thing.localArchiveExists(id) {
        guard let savedThing = NSKeyedUnarchiver.unarchiveObjectWithFile(Thing.archiveFilePath(id)) as? Thing else { return nil }
        self.id = savedThing.id
        self.name = savedThing.name
        self.image = savedThing.image
        self.imageUrl = savedThing.imageUrl
        super.init()
        print("Loaded Thing \(self.name) from archive")
    }
    else {
        guard let name: String = "name" <~~ json else { return nil}
        guard let imageUrl: NSURL = "url" <~~ json else { return nil}
        self.id = id
        self.name = name
        self.imageUrl = imageUrl
        super.init()
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)) {
            let data = NSData(contentsOfURL: imageUrl)
            dispatch_async(dispatch_get_main_queue(), {
                self.image = UIImage(data: data!)
                guard self.save() else {
                    print("ERROR. Could not archive")
                    return
                }
                print("Loaded Thing \(self.name) from server")
            })
        }
    }
}

Вот соответствующие части класса Thing:

// MARK: Properties
var id: Int?
var name: String
var imageUrl: NSURL?
var image: UIImage?

// MARK: Archiving Paths
static let DocumentsDirectory = NSFileManager().URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask).first!
static let ArchiveURL = DocumentsDirectory.URLByAppendingPathComponent("things")

// MARK: Types
struct PropertyKey {
    static let nameKey = "name"
    static let imageKey = "image"
    static let imageUrlKey = "imageUrl"
    static let idKey = "id"
}

// Returns the file URL at which a Thing with the given ID should be saved.
class func archiveFilePath(id: Int) -> String {
    return Thing.ArchiveURL.URLByAppendingPathComponent("thing\(id)").absoluteString
}

// Checks whether an archived copy of a Thing with the given ID exists.
class func localArchiveExists(id: Int) -> Bool {
    let fileManager = NSFileManager.defaultManager()
    return fileManager.fileExistsAtPath(Thing.archiveFilePath(id))
}

// MARK: NSCoding
func encodeWithCoder(coder: NSCoder) {
    coder.encodeObject(name, forKey: PropertyKey.nameKey)
    if image != nil {
        coder.encodeObject(image!, forKey: PropertyKey.imageKey)
    }
    if imageUrl != nil {
        coder.encodeObject(imageUrl!, forKey: PropertyKey.imageUrlKey)
    }
    coder.encodeInteger(id!, forKey: PropertyKey.idKey)
}

required convenience init?(coder aDecoder: NSCoder) {
    let name = aDecoder.decodeObjectForKey(PropertyKey.nameKey) as! String
    let image = aDecoder.decodeObjectForKey(PropertyKey.imageKey) as? UIImage
    let imageUrl = aDecoder.decodeObjectForKey(PropertyKey.imageUrlKey) as? NSURL
    let id = aDecoder.decodeIntegerForKey(PropertyKey.idKey) 

    // Must call designated initializer.
    self.init(name: name, image: image, imageUrl: imageUrl, id: id)
}

func save() -> Bool {        
    // For some reason I can't archive to file.
    return NSKeyedArchiver.archiveRootObject(self, toFile: Thing.archiveFilePath(self.id!))
}  

person cproctor    schedule 17.05.2016    source источник
comment
... предположительно Thing заявляет о своем соответствии NSCoding?   -  person Tommy    schedule 17.05.2016
comment
Проблема в структуре внутри вашего класса. Не соответствует кодировке   -  person Leo Dabus    schedule 17.05.2016
comment
@Tommy, да, Вещь объявлена: class Thing: NSObject, NSCoding, Glossy {   -  person cproctor    schedule 17.05.2016
comment
@LeoDabus, я попытался удалить структуру, и у меня все еще та же проблема. Не похоже, что структура должна быть проблемой, так как я ее не кодирую. Любые другие идеи? Вот полный класс, если это поможет. Спасибо!   -  person cproctor    schedule 17.05.2016


Ответы (1)


Я разобрался со своей проблемой: сохранение не удалось, потому что я еще не создал каталог, в котором пытался сохранить свою Вещь.

func save() -> Bool {        
    let archivedData = NSKeyedArchiver.archivedDataWithRootObject(self)
    do {
        try NSFileManager.defaultManager().createDirectoryAtURL(Thing.ArchiveURL, withIntermediateDirectories: true, attributes: [:])
        try archivedData.writeToFile(Thing.archiveFilePath(self.id!), options: [])
        return true
    } catch {
        print(error)
        return false
    }
}
person cproctor    schedule 20.05.2016