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

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

新規登録して質問してみよう
ただいま回答率
85.46%
Xcode

Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

Q&A

解決済

1回答

1340閲覧

Swift 構造体を配列にしたものを保存する方法について・tableview

samson66

総合スコア35

Xcode

Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

0グッド

0クリップ

投稿2020/08/29 05:12

編集2020/08/30 03:56

実現したいこと・やったこと
textfieldから入力した文字をtablviewにTODOリストの様に表示させ、
最終的には投稿時間やタグなど他の情報も扱えるように配列を含んだ構造体にし、
さらにuserDefaultsなどで保存機能を持たせようとコードを書き換えてみたのですが、
保存機能はついたものの一行だけしか保存されなかったりとうまくいかず、
他に先行例を検索してみても見つからなかったのでこちらで質問しました。

質問
以下がuserDefaultsを加える前の状態のコードなのですが、
textfieldから構造体の配列に入れた文字を保存する方法についてアドバイスお待ちしておりますm(_ _)m

import UIKit struct Plan : Codable{ let doing: String //イニシャライザ init(doing: String) { self.doing = doing } } class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate { //tableviewに表示させるスケジュールリスト var todo = [Plan]() @IBOutlet weak var textfield: UITextField! @IBOutlet weak var table: UITableView! override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view. table.dataSource = self table.delegate = self } @IBAction func add(_ sender: Any) { let mozi = textfield.text! todo.append(Plan(doing: mozi)) table.reloadData() } func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { return todo.count } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let plan = todo[indexPath.row] let cell = tableView.dequeueReusableCell(withIdentifier: "Cell",for: indexPath) cell.textLabel?.text = plan.doing return cell } }

8/30 追記
似たような質問の回答を下に自分なりに修正してみたのですが、またしても配列に一行しかappendされないという事象が起き、
保存はうまくいっても、配列として保存ができないという結果です。

import UIKit struct Plan : Codable{ let doing: String //イニシャライザ init(doing: String) { self.doing = doing } } class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate { //tableviewに表示させるスケジュールリスト var todo = [Plan]() var readArray = [Plan]() @IBOutlet weak var textfield: UITextField! @IBOutlet weak var table: UITableView! override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view. table.dataSource = self table.delegate = self readArray = Plan.readGoals() } @IBAction func add(_ sender: Any) { let mozi = textfield.text! //todo.append(Plan(doing: mozi)) var saveArray: [Plan] = [] saveArray.append(Plan(doing: mozi)) // goal配列をUserDefaultsに保存 Plan.saveGoals(Plan: saveArray) table.reloadData() print(readArray[0].doing) //ここまでは保存されているようです print(readArray[1].doing) //わずか2個目からThread 1: Fatal error: Index out of rangeという警告文がつきました。 print(readArray[2].doing) } func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { let readArray = Plan.readGoals() let aaa = readArray return aaa.count } func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { let plan = readArray[indexPath.row] let cell = tableView.dequeueReusableCell(withIdentifier: "Cell",for: indexPath) cell.textLabel?.text = plan.doing return cell } } extension Plan { static let userDefaultGoalKey = "goalSaveKey" static func saveGoals(Plan: [Plan]) { let defaults = UserDefaults.standard defaults.set(Plan.map{ $0.doing }, forKey: userDefaultGoalKey) defaults.synchronize() } static func readGoals() -> [Plan] { let defaults = UserDefaults.standard if let goalArray = defaults.object(forKey: userDefaultGoalKey) as? [String] { return goalArray.map{ Plan(doing: $0) } } return [] } }

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

UserDefaultsで配列を保存するだけでいいですかね?
投稿時間やタグなども構造体に追加すれば大丈夫だと思います。

private let todoListKey = "TodoListKey" var todo = [Plan]() override func viewDidLoad() { super.viewDidLoad() // 省略 // 画面表示時にUserDefaultsからTODOリストを取得 if let data = UserDefaults.standard.array(forKey: todoListKey) as? [Data] { self.todo = data.map { try! JSONDecoder().decode(Plan.self, from: $0) } } } @IBAction func add(_ sender: Any) { let mozi = textfield.text! todo.append(Plan(doing: mozi)) // ローカルの配列に追加してUserDefaultsに保存 // Data型で保存(Codable使っているので下記のようにエンコードできます) let data = self.todo.map { try? JSONEncoder().encode($0) } UserDefaults.standard.set(data, forKey: todoListKey) table.reloadData() }

投稿2020/08/29 09:29

編集2020/08/30 07:16
TakuyaAso

総合スコア1361

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

samson66

2020/08/30 03:49

回答ありがとうございます! 上記の方法で修正してみたのですが、テキストフィールドから文字を打ち込むと「[User Defaults] Attempt to set a non-property-list object ("プロジェクト名.Plan(doing: \"打ち込んだ文字\")"」というエラーが発生していましました。
TakuyaAso

2020/08/30 07:17

Data型でCodable使えるのでエンコードして保存,デコードするように変更しました
samson66

2020/08/30 16:30

ありがとうございます!無事に動きました! まだ保存の書き方に慣れていないので今回のソースをよく振り返ろうと思います。
TakuyaAso

2020/08/30 17:30

おお良かったです! サンプルとか簡単な検証にはUserDefaultsは便利ですが 実際はRealmなどのDataBaseに保存した方がいいので お時間あったら挑戦してみてくださいー
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.46%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問