7
16

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

【Swift5】Realmでデータを管理する

Last updated at Posted at 2020-03-01

はじめに

 メモアプリのタイトルのみをUserDefaultsで管理していたのですが、タグや日付などの情報も管理したくなり、UserDefaultsで保持すると複雑になりそうだったので、Realmへの移行を試みました!
 データの挿入まではわりとスムーズにできたのですが、autoIncrementや更新系の設定で、苦戦したので、その辺りを中心にまとめます!

環境

  • xcode 11.3.1
  • swift 5.1.3

Realmにデータを挿入する

 Realmにデータを挿入するには、Objectのモデルを作り、addという関数を使って保存してあげれば、データが挿入されるようです。

Memo.swift
import Foundation
import RealmSwift

class Memo: Object {
    @objc dynamic var title : String = ""
    @objc dynamic var content : String = ""
    @objc dynamic var tags : String = ""
    @objc dynamic var createdDate : Date = NSDate() as Date
    @objc dynamic var updatedDate : Date = NSDate() as Date
    
}

モデルは↑のように作りました!
データを挿入する際のコードはこんな感じです!

MemoTableViewController.swift(一部抜粋)
let memoModel = Memo()
let realm = try! Realm()

// オブジェクトに値をセットする
memoModel.title = memo
memoModel.content = memo
memoModel.tags = "test1"

// DBに書き込む
try! realm.write {
    realm.add(memoModel)
}

データを挿入するところまではこちらの記事を参考にしていただければ、わかりやすいと思います!
私もこちらを参考に実装しました!
[Realm][Swift4対応 完全保存版] 3.モデルオブジェクトの作成と書き込み

Realm内のデータを確認する

Realmのデータを確認する手順は次のようになります。

  1. Realm Browserをインストールする
  2. アプリを起動し、RealmのデータベースのPathを調べる
  3. ターミナルからopenコマンドで2のPathを開く

私はこの手順で動作確認できました!
スクリーンショット 2020-02-25 0.24.54.png

動作確認する方法はこちらの記事が参考になると思います!
【Swift】Realm BrowserでRealm Mobile Databaseの中身を確認する

idを自動連番(Auto Increment)で入れる

RealmはデフォルトでAuto Incrementするような機能がなさそう?だったので私はRObjectというクラスを作り、
そこにAuto Incrementを実装して、モデルで継承するように実装しました。

Memo.Swift
import Foundation
import RealmSwift

class Memo: RObject { // RObject←Objectに変更
    @objc dynamic var title : String = ""
    @objc dynamic var content : String = ""
    @objc dynamic var tags : String = ""
    @objc dynamic var createdDate : Date = NSDate() as Date
    @objc dynamic var updatedDate : Date = NSDate() as Date
    
}
RObject.swift
import Foundation
import Realm
import RealmSwift

class RObject: Object {
    // ID
    @objc dynamic var id = 0

    // データを保存。
    func save() {
        let realm = try! Realm()
        if realm.isInWriteTransaction {
            if self.id == 0 { self.id = self.createNewId() }
            realm.add(self, update: true)
        } else {
            try! realm.write {
                if self.id == 0 { self.id = self.createNewId() }
                realm.add(self, update: true)
            }
        }
    }

    // 新しいIDを採番します。
    private func createNewId() -> Int {
        let realm = try! Realm()
        return (realm.objects(type(of: self).self).sorted(byKeyPath: "id", ascending: false).first?.id ?? 0) + 1
    }

    // プライマリーキーの設定
    override static func primaryKey() -> String? {
        return "id"
    }
}
MemoTableViewController.swift(一部抜粋)
        if let selectedIndexPath = self.tableView.indexPathForSelectedRow{
            self.memos[selectedIndexPath.row] = memo
            
            // realm
            let realm = try! Realm()
            let memoModel = realm.objects(Memo.self).filter("id == " + String(sourceVC.memoId!)).first ?? Memo()
            try! realm.write {
                memoModel.title = memo
                memoModel.save() // realm.addから↑実装した.saveに変更
            }
        }else{
            self.memos.append(memo)
            // realm
            let memoModel = Memo()
            let realm = try! Realm()
            try! realm.write {
                memoModel.title = memo
                memoModel.content = memo
                memoModel.tags = "test1"
                memoModel.save()
            }

参考にした記事

データの更新をする

 ここまではわりとスムーズに行ったのですが、データ更新あたりから妻付き始めました。。
Realm().add(_:update:)を使ってデータを更新するという情報が多かったのですが、なぜかエラーに。。
色々調べた結果、次のように実装変えたらコンパイル通ったので参考にしてください。
おそらく、RealmSwiftの最新版だと
Realmのドキュメント

realm.add(self, update: true)
// ↓
realm.add(self, update: .modified)

元のテーブル(モデル)を削除する

 実装自体のエラーはなくなって自動連番の機能や主キーのIDが入れられるようになりました。
が、今度は起動時にIDがテーブルに存在しないというエラーが出ました。。
 調べたところ、テーブルの形式を変更したので、元々入っていたデータにIDがなく、エラーのようでした。。
対処方法は次の2つでした。

  • 既存のテーブル(Object)の削除
  • マイグレーション

まだ、不要なデータしかなかったのとマイグレーションが最初うまくできなかったので今回は、一旦既存テーブル削除を実施!

AppDelegate.swift
// 既存テーブルの削除
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
        let realmURL = Realm.Configuration.defaultConfiguration.fileURL!
        let realmURLs = [
            realmURL,
            realmURL.appendingPathExtension("lock"),
            realmURL.appendingPathExtension("note"),
            realmURL.appendingPathExtension("management")
        ]
        for URL in realmURLs {
            do {
                try FileManager.default.removeItem(at: URL)
            } catch {
                // handle error
            }
        }
  return true
}

 このコードを実装した状態で、アプリケーション起動するとテーブルが削除されるので、新しいモデルが挿入できるようになると思います!
 マイグレーションの方法に関しては、勉強して追記しようと思います😓

参考にした記事

まとめ

 今回はRealmでデータ管理がするのに必要な実装をしてみました。updateのあたりとかは特にBool型を引数に実装をしている記事が多かったので、要注意だと思います!
 また、swift周りは技術の進化が早く、どんどんバージョンが入れ替わり、参考にした記事のコードが動かないということがザラにあったので、書かれた時期と環境のバージョンが大事だなと思いました!
あと余談ですがswiftの独特のコーディングにも少し慣れてきた気がしました(segueなど)!笑

7
16
2

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
16

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?