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

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

ただいまの
回答率

90.61%

  • Swift

    7056questions

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

  • Xcode

    4004questions

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

  • iOS

    3908questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Realm

    195questions

    RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。

Relamでのエラー

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 211

haruka-t

score 77

class TopViewViewController: UIViewController,UITableViewDataSource,UITableViewDelegate {


    @IBOutlet weak var TopViewTabelView: UITableView!
    //var item = [String]()
    let ToDo = TopTodo()
    var todoItem: Results<TopTodo>!



    override func viewDidLoad() {
        super.viewDidLoad()

        // 永続化されているデータを取りだす
        do{
            let realm = try Realm()
            self.todoItem = realm.objects(TopTodo.self)
            self.TopViewTabelView.reloadData()
        }catch{

        }

        navigationController?.navigationBar.prefersLargeTitles = true //これがやりたかった設定やメモ帳のナビゲーションの可変のやつ
        navigationItem.title = "データベース"
        TopViewTabelView.tableFooterView = UIView(frame: .zero)
        self.view.backgroundColor = UIColor.rgb(r: 250, g: 204, b: 125, alpha: 1)
        TopViewTabelView.frame = CGRect(x: 0, y: 64, width: 375, height: 667)

    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        TopViewTabelView.reloadData()
    }

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


    @IBAction func addBtr(_ sender: Any) {
        let alert = UIAlertController(title: "タイトル", message: "メッセージ", preferredStyle: .alert)

        // OKボタンの設定
        let okAction = UIAlertAction(title: "OK", style: .default, handler: {
            (action:UIAlertAction!) -> Void in

            // OKを押した時入力されていたテキストを表示
            if let textFields = alert.textFields {

                // アラートに含まれるすべてのテキストフィールドを調べる
                for textField in textFields {
                    // 変数で持つ必要はないのではないでしょうか?
                    let newTodo = TopTodo()
                    newTodo.item = textField.text!
                    do{
                        let realm = try Realm()

                        // トランザクション開始
                        try realm.write({ () -> Void in
                            realm.add(newTodo)
                            print("ToDo Saved")
                        })
                        // ここでトランザクションが終了

                        // ここでtodoItemの更新
                        self.todoItem = realm.objects(TopTodo.self)
                        self.TopViewTabelView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
                        print(textField.text!)

                    }catch{
                        print("Save is Faild")
                    }
                }
                self.TopViewTabelView.reloadData()
            }
        })
        alert.addAction(okAction)

        // キャンセルボタンの設定
        let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil)
        alert.addAction(cancelAction)

        // テキストフィールドを追加
        alert.addTextField(configurationHandler: {(textField: UITextField!) -> Void in
            textField.placeholder = "テキスト"
        })
        alert.view.setNeedsLayout() // シミュレータの種類によっては、これがないと警告が発生

        self.present(alert, animated: true, completion: nil)
    }


    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return todoItem.count

    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let Cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)
        /*let now = NSDate() // 現在日時の取得

        let dateFormatter = DateFormatter()
        dateFormatter.locale = NSLocale(localeIdentifier: "en_US") as Locale? // ロケールの設定
        dateFormatter.dateFormat = "yyyy年MM月dd日 HH:mm"//:ss" // 日付フォーマットの設定

        let deterstring = dateFormatter.string(from: now as Date)
        print(deterstring) // -> 2014/06/25 02:13:18*/

        let object = todoItem[indexPath.row]
        Cell.textLabel?.text = object.item //+ deterstring
        Cell.detailTextLabel?.text = object.date.description
        //Cell.detailTextLabel?.text = deterstring

        return Cell
    }

    func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) {


        if(editingStyle == UITableViewCellEditingStyle.delete) {
            do{
                let realm = try Realm()
                try realm.write {
                    realm.delete(self.todoItem[indexPath.row])
                }
                TopViewTabelView.deleteRows(at: [indexPath], with: UITableViewRowAnimation.fade)
            }catch{
            }
            TopViewTabelView.reloadData()
        }
    }

}
import RealmSwift

class TopTodo: Object{
    // 管理用 ID。プライマリーキー
    @objc dynamic var id = 0
    @objc dynamic  var item = ""

    /// 最終更新日時
    @objc dynamic var date = NSDate()

     //id をプライマリーキーとして設定
    override static func primaryKey() -> String? {
        return "id"
    }

}

 困っていること

セルのサブタイトルの更新時間を取得したく、オブジェクトクラスに@objc dynamic var date = NSDate()を追加したところ、numberOfRowsInSectionreturn todoItem.counでエラーが起きてしまった。
以下がエラーの内容です。
Fatal error: Unexpectedly found nil while unwrapping an Optional value
実行すると落ちてしまいます。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+1

その行で Unexpectedly found nil while unwrapping an Optional valueで落ちたと言うことは、todoItemがnilだったということです。

todoItemを設定している処理は

        do{
            let realm = try Realm()
            self.todoItem = realm.objects(TopTodo.self)
            self.TopViewTabelView.reloadData()
        }catch{

        }


です。
おそらくtodoItemがnilになった原因は、try Realm()の処理がエラーになり、self.todoItem = 〜の処理が実行されずにcatch節に実行が移ったのだと思います。
最悪なのは、do-catchの構文を使っているのに、catch節の中に何も書いていないことです。このため、ここでエラーが発生しているのにエラー処理を何もせず、エラーがなかったことにされ、todoItemがnilのまま続きが動作して質問の箇所でエラーが発生したのだと思います。
せめて、次のようにcatch節でエラーをprintしておけば、try Realm()がエラーになった原因がコンソールメッセージで出力されるはずです。

        }catch{
            print(error)
        }

そもそも try Realm()がエラーになったら、Realmが一切使えないということなので続行しても仕方がなく、この段階でエラーになった原因をしっかり調べる必要があります。
そのため、通常ここではdo-catchを使わず、次のように try!を使って、
エラー発生時はエラーメッセージを表示してすぐにアプリを停止させるのが一般的だと思います。

    let realm = try! Realm()
    self.todoItem = realm.objects(TopTodo.self)

try Realm()がエラーになった原因については、そのエラーメッセージを確認しないと正確なことはわかりませんが、上記の修正をして表示されたエラーメッセージにMigration is requiredといったようなメッセージが出ていたら、

オブジェクトクラスに@objc dynamic var date = NSDate()を追加した

ということなので、前回実行した時とデータ構造が変わり、前回のアプリ実行で保存していたデータが使用できなくてエラーになったということだと思います。

このアプリがまだユーザーにリリースされていないものであれば、今自分がテストに使っているシミュレータや実機からこのアプリを削除して再実行するだけで以前に保存していたデータは削除されるので、エラーは解消されると思います。

もし、既にこのアプリがユーザーにリリースされていて、今回はそのアプリのバージョンアップでデータ項目を追加しようとしているのなら、ユーザーが保存していたデータを引き継いで新しいデータ構造で処理できるよう「マイグレーション」という処理を実施する必要があります。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/06/25 13:53

    懇切にご教授していただき有難うございます。
    解決できました。
    今回はまだ開発中の段階でしたのでマイグレーションを行う必要はありませんでした。
    しかし、マイグレーションという処理は今後必要になっていくと思いますので勉強したいと思います。

    キャンセル

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

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

関連した質問

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

  • Swift

    7056questions

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

  • Xcode

    4004questions

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

  • iOS

    3908questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Realm

    195questions

    RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。