Сохранить пользовательские объекты в NSUserDefaults

У меня новости ViewController и TeamViewController. TeamViewController содержит tableView TeamObjects, который при выборе добавляется в массив. Я хочу добавить этот массив в NSUserDefaults, чтобы я мог получить к ним доступ из NewsController, которые содержат запрос URL-адреса, где необходимы teamObjects. Однако я продолжаю получать:

«Попытка вставить объект списка, не являющийся свойством («») для ключевых команд»

Я открыт для других предложений, если есть лучшие способы, чем хранить его в NSUserDefaults

didSelectRowAtIndexPath метод

override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
    tableView.deselectRowAtIndexPath(indexPath, animated: true)


    let team = self.teamArray[indexPath.row] as Team
    var removed = false

    for (index, value) in enumerate(self.teamSelected) {
        if (value == team) {
            self.teamSelected.removeAtIndex(index)
            removed = true
        }
    }

    if (!removed) {
        self.teamSelected.append(team)
    }

    var userDefaults = NSUserDefaults.standardUserDefaults()
    userDefaults.setValue(self.teamSelected, forKey: "teams")
    userDefaults.synchronize()

    tableView.reloadData()
}

Мой объект

class Team: NSObject{
    var id: Int!
    var name: NSString!
    var shortname: NSString!


    init(id: Int, name:NSString, shortname: NSString) {
        self.id = id
        self.name = name
        self.shortname = shortname

    }

}

person Peter Pik    schedule 01.05.2015    source источник


Ответы (2)


На самом деле вам нужно будет заархивировать пользовательский объект в NSData, затем сохранить его в пользовательских настройках по умолчанию, извлечь из пользовательских настроек по умолчанию и снова разархивировать. Вы можете заархивировать это так

let teams = [Team(id: 1, name: "team1", shortname: "t1"), Team(id: 2, name: "team2", shortname: "t2")]

var userDefaults = UserDefaults.standard
let encodedData: Data = NSKeyedArchiver.archivedData(withRootObject: teams)
userDefaults.set(encodedData, forKey: "teams")
userDefaults.synchronize()

и разархивировать вот так

let decoded  = userDefaults.data(forKey: "teams")
let decodedTeams = NSKeyedUnarchiver.unarchiveObject(with: decoded) as! [Team]

Но если вы только что сделали это, вы получите

.Team encodeWithCoder:]: unrecognized selector sent to instance

Вам нужно будет заставить команду соответствовать NSCoding так же, как это

class Team: NSObject, NSCoding {
    var id: Int
    var name: String
    var shortname: String


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

    }

    required convenience init(coder aDecoder: NSCoder) {
        let id = aDecoder.decodeInteger(forKey: "id")
        let name = aDecoder.decodeObject(forKey: "name") as! String
        let shortname = aDecoder.decodeObject(forKey: "shortname") as! String
        self.init(id: id, name: name, shortname: shortname)
    }

    func encode(with aCoder: NSCoder) {
        aCoder.encode(id, forKey: "id")
        aCoder.encode(name, forKey: "name")
        aCoder.encode(shortname, forKey: "shortname")
    }
}
person mohamede1945    schedule 01.05.2015
comment
Как вы справляетесь с ошибкой, вызванной этой строкой let decodedTeams = NSKeyedUnarchiver.unarchiveObjectWithData(decoded) as! [Team] (ошибка Can throw but the error is not handled)? - person afkatja; 09.11.2016
comment
@afkatja Проверьте обновленный код - теперь он должен работать :) - person Fluidity; 15.11.2016
comment
В частности, спасибо за включение decodeInteger в этот пример, благодаря которому я заметил, что приведение типов действительно имеет значение в этой функции. - person Tamás Sengel; 19.06.2017
comment
мы можем сохранить UIImage с помощью этого метода? - person Talha Ahmad Khan; 03.12.2017
comment
Вы должны преобразовать его в NSData. Вы не можете сохранить UIImage напрямую. - person mohamede1945; 03.12.2017
comment
необходимое удобство init (coder aDecoder: NSCoder) этот метод не вызывается для меня, любая помощь приветствуется - person kalpa; 16.03.2018
comment
Интересно, можно ли это сделать с помощью структуры? - person Sentry.co; 19.09.2018
comment
что, если внутри команды есть пользовательский объект, нужно ли нам также добавлять NSCoding в класс этого настраиваемого объекта?? - person Ankit Kumar Gupta; 14.03.2019

Вы можете попробовать NSKeyedUnarchiver, как показано ниже.

Здесь я сохранил объект UIColor в UserDefault. Вы можете изменить его с помощью своего объекта.

 NSData *colorData = [NSKeyedArchiver archivedDataWithRootObject:color];
[[NSUserDefaults standardUserDefaults] setObject:colorData forKey:@"myColor"];

и восстановить это

NSData *colorData = [[NSUserDefaults standardUserDefaults] objectForKey:@"myColor"];
UIColor *color = [NSKeyedUnarchiver unarchiveObjectWithData:colorData];

Примечание: это Objective C, но я надеюсь, что вы сможете скрыть это в Swift

Надеюсь, это решит вашу проблему


Обновление: Swift 5

Архив в Swift

do{
    let colorAsData = try NSKeyedArchiver.archivedData(withRootObject: UIColor.red, requiringSecureCoding: true)
    UserDefaults.standard.set(colorAsData, forKey: "myColor")
    UserDefaults.standard.synchronize()
}catch (let error){ 
    #if DEBUG
        print("Failed to convert UIColor to Data : \(error.localizedDescription)")
    #endif
}

Разархивировать в Swift

do{
    if let colorAsData = UserDefaults.standard.object(forKey: "myColor") as? Data{
        if let color = try NSKeyedUnarchiver.unarchivedObject(ofClasses: [UIColor.self], from: colorAsData){
            // Use Color
        }
    }
}catch (let error){
    #if DEBUG
        print("Failed to convert UIColor to Data : \(error.localizedDescription)")
    #endif
}
person Malav Soni    schedule 01.05.2015
comment
из-за необработанного исключения «NSInvalidArgumentException», причина: «-[Striky.Team encodeWithCoder:]: нераспознанный селектор - person Peter Pik; 01.05.2015
comment
Это потому, что вам нужно соответствовать протоколу NSCoding. - person Steve; 13.04.2016