web-dev-qa-db-ja.com

Swift 3にカスタムオブジェクトを保存しようとしたときに、プロパティリスト以外のオブジェクトを挿入しようとしました

NSCodingプロトコルに準拠する単純なオブジェクトがあります。

import Foundation

class JobCategory: NSObject, NSCoding {
    var id: Int
    var name: String
    var URLString: String

    init(id: Int, name: String, URLString: String) {
        self.id = id
        self.name = name
        self.URLString = URLString
    }

    // MARK: - NSCoding
    required init(coder aDecoder: NSCoder) {
        id = aDecoder.decodeObject(forKey: "id") as? Int ?? aDecoder.decodeInteger(forKey: "id")
        name = aDecoder.decodeObject(forKey: "name") as! String
        URLString = aDecoder.decodeObject(forKey: "URLString") as! String
    }

    func encode(with aCoder: NSCoder) {
        aCoder.encode(id, forKey: "id")
        aCoder.encode(name, forKey: "name")
        aCoder.encode(URLString, forKey: "URLString")
    }
}

インスタンスをUserDefaultsに保存しようとしていますが、次のエラーで失敗し続けます。

キャッチされない例外 'NSInvalidArgumentException'によるアプリの終了、理由: 'キーjobCategoryの非プロパティリストオブジェクトを挿入しようとしています'

これは、UserDefaultsに保存するコードです。

enum UserDefaultsKeys: String {
    case jobCategory
}

class ViewController: UIViewController {

    @IBAction func didTapSaveButton(_ sender: UIButton) {
        let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")

        let userDefaults = UserDefaults.standard
        userDefaults.set(category, forKey: UserDefaultsKeys.jobCategory.rawValue)
        userDefaults.synchronize()
    }
}

キーの列挙値を通常の文字列に置き換えましたが、同じエラーが引き続き発生します。何がこれを引き起こしているのでしょうか?

34
Isuru

NSKeyedArchiver.archivedData(withRootObject:) を使用してDataインスタンスからJobCategoryインスタンスを作成し、 NSKeyedUnarchiver.unarchiveObject(with:) を使用してDataインスタンスをUserDefaults以降に展開する必要があります。このようにしてみてください。

UserDefaultsにデータを保存する場合

let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")
let encodedData = NSKeyedArchiver.archivedData(withRootObject: category)
let userDefaults = UserDefaults.standard
userDefaults.set(encodedData, forKey: UserDefaultsKeys.jobCategory.rawValue)

UserDefaultsからデータを取得する場合

let decoded  = UserDefaults.standard.object(forKey: UserDefaultsKeys.jobCategory.rawValue) as! Data
let decodedTeams = NSKeyedUnarchiver.unarchiveObject(with: decoded) as! JobCategory
print(decodedTeams.name)
70
Nirav D

pdate Swift 4、Xcode 1

簡単にアクセスできるように、その周りに構造体を作成しました。

//set, get & remove User own profile in cache
struct UserProfileCache {
    static let key = "userProfileCache"
    static func save(_ value: Profile!) {
         UserDefaults.standard.set(try? PropertyListEncoder().encode(value), forKey: key)
    }
    static func get() -> Profile! {
        var userData: Profile!
        if let data = UserDefaults.standard.value(forKey: key) as? Data {
            userData = try? PropertyListDecoder().decode(Profile.self, from: data)
            return userData!
        } else {
            return userData
        }
    }
    static func remove() {
        UserDefaults.standard.removeObject(forKey: key)
    }
}

プロファイルは、Jsonでエンコードされたオブジェクトです。

struct Profile: Codable {
let id: Int!
let firstName: String
let dob: String!
}

使用法:

//save details in user defaults...
UserProfileCache.save(profileDetails)

それが役立つことを願っています!!!

ありがとう

11
Harjot Singh

辞書をユーザーデフォルトに保存

let data = NSKeyedArchiver.archivedData(withRootObject: DictionaryData)
UserDefaults.standard.set(data, forKey: kUserData)

辞書を取得する

let outData = UserDefaults.standard.data(forKey: kUserData)
let dict = NSKeyedUnarchiver.unarchiveObject(with: outData!)as!NSDictionary
4
Niraj Paul