質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.03%

Swift3 UserDefaultに配列が保存されない

解決済

回答 3

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,231

Tats.N

score 56

こんにちは。いつもありがとうございます。

UserDefaultに配列を保存してアプリを閉じてもデータが残るようにしたいのです。
下のaddItem関数内で保存しているのですが、アプリを閉じると消えてしまいます。何を間違ってますか?
わかる方おられましたらご教授ください。宜しくお願いします。

var items = [[String](),[String](),[String]()]

    let ud = [UserDefaults.standard,UserDefaults.standard,UserDefaults.standard]


 @IBAction func addItem(_ sender: Any) {

//略

        // save into memory
        ud[fridgeSegOutlet.selectedSegmentIndex].set(items[fridgeSegOutlet.selectedSegmentIndex], forKey: String(fridgeSegOutlet.selectedSegmentIndex))

        self.textInput.resignFirstResponder()
    }

全体のコードです

import UIKit

class ViewController: UIViewController, UITableViewDelegate, UITableViewDataSource {

    @IBOutlet var textInput: UITextField!
    @IBOutlet var table: UITableView!


    //var fridgeItem = [String]() // this is going to init this array so i don't need to put anything in it.
    //var freezerItem = [String]()
    //var otherItem = [String]()
    var items = [[String](),[String](),[String]()]
    //Fridge, Freezer, Other

    //refered:http://cabbalog.blogspot.com/2016/09/xcode8-swift3-nsuserdefaults.html
    let ud = [UserDefaults.standard,UserDefaults.standard,UserDefaults.standard]

    @IBOutlet var fridgeSegOutlet: UISegmentedControl!
    @IBAction func fridgeSeg(_ sender: Any) {
        //if ud[fridgeSegOutlet.selectedSegmentIndex].object(forKey: String(fridgeSegOutlet.selectedSegmentIndex)) != nil{
            //items[fridgeSegOutlet.selectedSegmentIndex] = ud[fridgeSegOutlet.selectedSegmentIndex].object(forKey: String(fridgeSegOutlet.selectedSegmentIndex)) as! [String]
        //}


        table.reloadData()
    }

    @IBAction func addItem(_ sender: Any) {
        // Add item to list
        if textInput.text != "" {
            items[fridgeSegOutlet.selectedSegmentIndex].append(textInput.text!)
        }
        // Clear text input field
        textInput.text = ""

        // refresh table view
        table.reloadData()

        // save into memory
        ud[fridgeSegOutlet.selectedSegmentIndex].set(items[fridgeSegOutlet.selectedSegmentIndex], forKey: String(fridgeSegOutlet.selectedSegmentIndex))
        self.textInput.resignFirstResponder()
    }

    override func viewDidLoad() {
        super.viewDidLoad()
        if ud[fridgeSegOutlet.selectedSegmentIndex].object(forKey: String(fridgeSegOutlet.selectedSegmentIndex)) != nil{
            //items[fridgeSegOutlet.selectedSegmentIndex] = ud[fridgeSegOutlet.selectedSegmentIndex].object(forKey: String(fridgeSegOutlet.selectedSegmentIndex)) as! [String]
        }
        //for index in 0...2{
        //    ud[index].set(items[index], forKey: String(index))
        //}
    }

    public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int{

        return items[fridgeSegOutlet.selectedSegmentIndex].count
    }

    public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell{

        let cell = UITableViewCell(style: .default, reuseIdentifier: "Cell")

        cell.textLabel?.text = items[fridgeSegOutlet.selectedSegmentIndex][indexPath.row]

        return cell

    }

    func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) {
        if editingStyle == UITableViewCellEditingStyle.delete{
            items[fridgeSegOutlet.selectedSegmentIndex].remove(at: indexPath.row)

            table.deleteRows(at: [indexPath], with: .automatic)

            ud[fridgeSegOutlet.selectedSegmentIndex].object(forKey: String(fridgeSegOutlet.selectedSegmentIndex))
        }
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.view.endEditing(true)
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }


}
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 3

checkベストアンサー

0

保存したデータを読み込んでいないようにみえるのですが?

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/12/31 09:31

    超凡ミスでした。。ありがとうございます。

    キャンセル

0

下記を追加するとどうでしょうか?

ud.synchronize()

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/12/16 14:14

    // save into memory
    ud[fridgeSegOutlet.selectedSegmentIndex].set(items[fridgeSegOutlet.selectedSegmentIndex], forKey: "\(fridgeSegOutlet.selectedSegmentIndex)")
    ud[fridgeSegOutlet.selectedSegmentIndex].synchronize()

    このようにして見ましたが、やはり保存されてなかったです。。

    キャンセル

  • 2016/12/16 14:28 編集

    var items = [[String](),[String](),[String]()]

    let ud = [UserDefaults.standard,UserDefaults.standard,UserDefaults.standard]


    @IBAction func addItem(_ sender: Any) {

    //略

    // save into memory
    ud[fridgeSegOutlet.selectedSegmentIndex].set(items[fridgeSegOutlet.selectedSegmentIndex], forKey: String(fridgeSegOutlet.selectedSegmentIndex))
    ud[fridgeSegOutlet.selectedSegmentIndex].synchronize()
    self.textInput.resignFirstResponder()
    }


    でどうですか?

    キャンセル

  • 2016/12/16 14:34

    こちらも参考にしてみてください。
    http://dev.classmethod.jp/smartphone/swift-3-0-userdefaults/

    キャンセル

  • 2016/12/16 18:35

    `ud.synchronize()`の戻り値がtrueになっていますか?

    キャンセル

0

UserDefaults.standard は Singletonなので配列にしている時点で何か勘違いをされているようです。3つのセグメンテッドコントロールの状態を保存するだけなら配列で保存すれば良いと思います。

import UIKit

class ViewController: UIViewController {

    var items:Array<Int> = [0,0,0]

    let ud = UserDefaults.standard

    override func viewDidLoad() {
        super.viewDidLoad()

        ud.register(defaults: ["items":items])
        ud.synchronize()

        print(ud.array(forKey: "items"))
        // Optional([0, 0, 0]) 1回目
        // Optional([0, 1, 0]) 2回目

        items[1] = 1
        ud.set(items,forKey:"items")
        ud.synchronize()

        print(ud.array(forKey: "items"))
        // Optional([0, 1, 0]) 1回目
        // Optional([0, 1, 0]) 2回目

 }

}


Dictionaryでいくならこんな感じ

import UIKit

class ViewController: UIViewController {

    var items:Dictionary<String,Any> = ["seg1":0,"seg2":0,"seg3":0]

    let ud = UserDefaults.standard

    override func viewDidLoad() {
        super.viewDidLoad()

        ud.register(defaults: ["items":items])
        ud.synchronize()

        print(ud.dictionary(forKey: "items"))
        // Optional(["seg3": 0, "seg2": 0, "seg1": 0]) 1回目
        // Optional(["seg3": 0, "seg2": 1, "seg1": 0]) 2回目

        items["seg2"] = 1
        ud.set(items,forKey: "items")
        ud.synchronize()

        print(ud.dictionary(forKey: "items"))
        // Optional(["seg3": 0, "seg2": 1, "seg1": 0]) 1回目
        // Optional(["seg3": 0, "seg2": 1, "seg1": 0]) 2回目

        }

}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.03%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る