jak zapisywać i czytać tablicę tablicy w NSUserdefaults w swift?

81

Muszę utworzyć tablicę, aby dodać obiekty w tym formacie, takie jak słownik w języku Swift: ["klucz1": "wartość1", "klucz2": "wartość2"]

Kiedy próbuję go zapisać NSUserDefaultswszystko jest poprawne, ale gdy czytam NSUserDefaultskluczem to się zawiesza. Jakiego typu danych potrzebuje mój var obj?

let def = NSUserDefaults.standardUserDefaults()
var key = "keySave"
var element: AnyObject!

var array1: [AnyObject!] = []
array1.append(["key1": "val1", "key2": "val2"])
array1.append(["key1": "val1", "key2": "val2"])

//save
var savestring : [AnyObject!]
savestring = array1
var defaults = NSUserDefaults.standardUserDefaults()
defaults.setObject(savestring, forKey: key)
defaults.synchronize()

//read
var obj: [AnyObject!] = []
if(obj != nil){
    print("size: ")
    print(obj.count) //vary long value confused..
    element = obj[0]  //crash
    print(element.objectForKey("key1"))
}
user3745888
źródło
3
Wygląda na to, że NSUserDefaults może obsłużyć String, ale nie Array i Dictionary, działając tylko dla typów NS * opisujących dokumenty. Więc to wygląda na błąd w Swift.
zaph
Aby wyjść trochę poza to, co mówi @Zaph, testowałem z [String] i nie działa, ale działa z [NSString]. Sprawdź moją odpowiedź.
HpTerm

Odpowiedzi:

215

Pytanie brzmi „tablica tablic”, ale myślę, że większość ludzi prawdopodobnie przychodzi tutaj, chcąc się dowiedzieć, jak zapisać tablicę w UserDefaults . Dla tych osób dodam kilka typowych przykładów.

Tablica ciągów

Zapisz tablicę

let array = ["horse", "cow", "camel", "sheep", "goat"]

let defaults = UserDefaults.standard
defaults.set(array, forKey: "SavedStringArray")

Pobierz tablicę

let defaults = UserDefaults.standard
let myarray = defaults.stringArray(forKey: "SavedStringArray") ?? [String]()

Tablica int

Zapisz tablicę

let array = [15, 33, 36, 723, 77, 4]

let defaults = UserDefaults.standard
defaults.set(array, forKey: "SavedIntArray")

Pobierz tablicę

let defaults = UserDefaults.standard
let array = defaults.array(forKey: "SavedIntArray")  as? [Int] ?? [Int]()

Tablica Bool

Zapisz tablicę

let array = [true, true, false, true, false]

let defaults = UserDefaults.standard
defaults.set(array, forKey: "SavedBoolArray")

Pobierz tablicę

let defaults = UserDefaults.standard
let array = defaults.array(forKey: "SavedBoolArray")  as? [Bool] ?? [Bool]()

Tablica dat

Zapisz tablicę

let array = [Date(), Date(), Date(), Date()]

let defaults = UserDefaults.standard
defaults.set(array, forKey: "SavedDateArray")

Pobierz tablicę

let defaults = UserDefaults.standard
let array = defaults.array(forKey: "SavedDateArray")  as? [Date] ?? [Date]()

Tablica obiektów

Zapisywanie obiektów niestandardowych (a co za tym idzie tablic obiektów) wymaga trochę więcej pracy UserDefaults. Zobacz poniższe łącza, aby dowiedzieć się, jak to zrobić.

Uwagi

  • Operator łączenia nil ( ??) umożliwia zwrócenie zapisanej tablicy lub pustej tablicy bez awarii. Oznacza to, że jeśli obiekt zwróci nil, to ??zamiast tego zostanie użyta wartość następująca po operatorze.
  • Jak widać, podstawowa konfiguracja była taka sama dla Int, Booli Date. Przetestowałem to również z Double. O ile wiem, wszystko, co możesz zapisać na liście właściwości, będzie działać w ten sposób.
Suragch
źródło
16

Wystarczy dodać to, co @Zaph mówi w komentarzach.

Mam ten sam problem co ty, jak wiesz, tablica Stringnie jest zapisana. Mimo że Apple łączy typy takie jak String i NSString, nie byłem w stanie zapisać tablicy [String]żadnego z nich [AnyObject].

Jednak szereg [NSString]działa dla mnie.

Twój kod może wyglądać tak:

var key = "keySave"

var array1: [NSString] = [NSString]()
array1.append("value 1")
array1.append("value 2")

//save
var defaults = NSUserDefaults.standardUserDefaults()
defaults.setObject(array1, forKey: key)
defaults.synchronize()

//read
if let testArray : AnyObject? = defaults.objectForKey(key) {
    var readArray : [NSString] = testArray! as [NSString]
}

Zauważ, że utworzyłem tablicę NSString, a nie słownik. Nie sprawdzałem, czy działa ze słownikiem, ale prawdopodobnie będziesz musiał zdefiniować rzeczy, [NSString : NSString]aby działał.

EDYTOWAĆ

Ponownie czytając twoje pytanie i tytuł, o którym mówisz array of array. Myślę, że tak długo, jak zostaniesz z NSString, tablica będzie działać. Jeśli jednak uważasz, że moja odpowiedź jest nieistotna, daj mi znać w komentarzach, a usunę ją.

HpTerm
źródło
11

Oto przykład czytania i pisania listy obiektów typu, SNStockktóry implementuje NSCoding- mamy akcesor do całej listy watchlistoraz dwie metody dodawania i usuwania obiektów, czyli addStock(stock: SNStock)i removeStock(stock: SNStock).

import Foundation

class DWWatchlistController {

  private let kNSUserDefaultsWatchlistKey: String = "dw_watchlist_key"

  private let userDefaults: NSUserDefaults

  private(set) var watchlist:[SNStock] {

    get {
      if let watchlistData : AnyObject = userDefaults.objectForKey(kNSUserDefaultsWatchlistKey) {
        if let watchlist : AnyObject = NSKeyedUnarchiver.unarchiveObjectWithData(watchlistData as! NSData) {
          return watchlist as! [SNStock]
        }
      }
      return []
    }

    set(watchlist) {
      let watchlistData = NSKeyedArchiver.archivedDataWithRootObject(watchlist)
      userDefaults.setObject(watchlistData, forKey: kNSUserDefaultsWatchlistKey)
      userDefaults.synchronize()
    }
  }

  init() {
    userDefaults = NSUserDefaults.standardUserDefaults()
  }

  func addStock(stock: SNStock) {
    var watchlist = self.watchlist
    watchlist.append(stock)
    self.watchlist = watchlist
  }

  func removeStock(stock: SNStock) {
    var watchlist = self.watchlist
    if let index = find(watchlist, stock) {
      watchlist.removeAtIndex(index)
      self.watchlist = watchlist
    }
  }

}

Pamiętaj, że twój obiekt musi zostać zaimplementowany,NSCoding inaczej kodowanie nie zadziała. Oto jak SNStockwygląda:

import Foundation

class SNStock: NSObject, NSCoding
{
  let ticker: NSString
  let name: NSString

  init(ticker: NSString, name: NSString)
  {
    self.ticker = ticker
    self.name = name
  }

  //MARK: NSCoding

  required init(coder aDecoder: NSCoder) {
    self.ticker = aDecoder.decodeObjectForKey("ticker") as! NSString
    self.name = aDecoder.decodeObjectForKey("name") as! NSString
  }

  func encodeWithCoder(aCoder: NSCoder) {
    aCoder.encodeObject(ticker, forKey: "ticker")
    aCoder.encodeObject(name, forKey: "name")
  }

  //MARK: NSObjectProtocol

  override func isEqual(object: AnyObject?) -> Bool {
    if let object = object as? SNStock {
      return self.ticker == object.ticker &&
        self.name == object.name
    } else {
      return false
    }
  }

  override var hash: Int {
    return ticker.hashValue
  }
}

Mam nadzieję że to pomoże!

Zorayr
źródło
3

Spróbuj tego.

Aby uzyskać dane z UserDefaults.

var defaults = NSUserDefaults.standardUserDefaults()
var dict : NSDictionary = ["key":"value"]
var array1: NSArray = dict.allValues // Create a dictionary and assign that to this array
defaults.setObject(array1, forkey : "MyKey")

var myarray : NSArray = defaults.objectForKey("MyKey") as NSArray
println(myarray)
Alvin Varghese
źródło
Dodam var dict: [AnyObject!] = [] I var array1: NSArray = dict, ale to jest puste. Jak to zrobić poprawnie?
user3745888
@AlvinVarghese Właściwie wypróbuj kod. Pojawiła się literówka i nawet jeśli zostanie to naprawione, jedyną zachowaną rzeczą jest tablica z jednym wpisem: „wartość”.
zaph
3

Swift 4.0

Sklep:

let arrayFruit = ["Apple","Banana","Orange","Grapes","Watermelon"]    

 //store in user default
 UserDefaults.standard.set(arrayFruit, forKey: "arrayFruit")

Sprowadzać:

if let arr = UserDefaults.standard.array(forKey: "arrayFruit") as? [String]{
        print(arr)
    }
Panie Javed Multani
źródło
1

Tutaj jest:

var array : [String] = ["One", "Two", "Three"]
let userDefault = UserDefaults.standard
// set
userDefault.set(array, forKey: "array")
// retrieve 
if let fetchArray = userDefault.array(forKey: "array") as? [String] {
    // code 
}
Andrei Levchenko
źródło
0

Jeśli pracujesz z Swift 5+ , musisz używać UserDefaults.standard.setValue(value, forKey: key)i uzyskać zapisane dane, których musisz użyć UserDefaults.standard.dictionary(forKey: key).

Wimukthi Rajapaksha
źródło