-->

2020年5月3日日曜日

JSONデータを取り込んでRealmに保存する

プロジェクト内のJSONデータをRealmのモデルに変換して保存する。
xmlやplistを使うことも考えたが、codableを使うことで独自のモデルにも容易に対応させることができた。

環境


  • Xcode 11.4.1
  • Swift 5
  • iOS 13.4.1

StringやIntなどは自動Codableに準拠しているので、encodeやdecodeメソッドを使う必要はないが、
今回はその他のデータ型にも対応できるようにencode、decodeメソッドを実装した。

モデル[Recipe]

  1. import Foundation
  2. import RealmSwift
  3.  
  4. class Recipe: Object,Decodable {
  5. @objc dynamic var name = ""
  6. @objc dynamic var category = ""
  7. @objc dynamic var country = ""
  8. @objc dynamic var season = ""
  9. var ingredients = List<Ingredient>() //RecipeとIngredientは1対多
  10. override static func primaryKey() -> String? {
  11. return "name"
  12. }
  13. private enum CodingKeys:String,CodingKey{
  14. case name
  15. case category
  16. case country
  17. case season
  18. case ingredients
  19. }
  20. required convenience public init(from decoder:Decoder) throws{
  21. self.init()
  22. let container = try decoder.container(keyedBy: CodingKeys.self)
  23.  
  24. name = try container.decode(String.self, forKey: .name)
  25. category = try container.decode(String.self, forKey: .category)
  26. country = try container.decode(String.self, forKey: .country)
  27. season = try container.decode(String.self, forKey: .season)
  28. //一度配列に格納した後、Listに変換する
  29. let ingredientsArray = try container.decode([Ingredient].self, forKey: .ingredients)
  30. ingredients = ingredientsArray.reduce(List<Ingredient>()) {$0.append($1); return $0}
  31. }
  32. }

モデル[ingredient]

  1. import Foundation
  2. import RealmSwift
  3.  
  4. class Ingredient: Object,Decodable {
  5. @objc dynamic var name = ""
  6. let recipes = LinkingObjects(fromType: Recipe.self, property: "ingredients")
  7. private enum CodingKeys: String, CodingKey {
  8. case name
  9. }
  10. required convenience public init(from decoder: Decoder) throws {
  11. self.init()
  12. let container = try decoder.container(keyedBy: CodingKeys.self)
  13.  
  14. name = try container.decode(String.self, forKey: .name)
  15.  
  16. }
  17. }

JSONファイル

[
        {
        "name": "トマトパスタ",
        "category": "パスタ",
        "country": "イタリアン",
        "season": "Any",
        "ingredients":[
            {
                "name": "トマト"
            },
            {
                "name": "にんにく"
            }
        ]
        },
        {
        "name": "カツ丼",
        "category": "丼ぶり",
        "country": "和食",
        "season": "Any",
        "ingredients":[
            {
                "name": "たまねぎ"
            },
            {
                "name": "卵"
            },
            {
                "name": "豚肉"
            }
        ]
        },
        {
        "name": "ペペロンチーノ",
        "category": "パスタ",
        "country": "イタリアン",
        "season": "Any",
        "ingredients":[
            {
                "name": "にんにく"
            },
            {
                "name": "鷹の爪"
            }
        ]
        },
        {
        "name": "カレー",
        "category": "カレー",
        "country": "和食",
        "season": "Any",
        "ingredients":[
            {
                "name": "人参"
            },
            {
                "name": "たまねぎ"
            },
            {
                "name": "じゃがいも"
            },
            {
                "name": "牛肉"
            }
        ]
        },
        {
        "name": "シチュー",
        "category": "シチュー",
        "country": "和食",
        "season": "Any",
        "ingredients":[
            {
                "name": "人参"
            },
            {
                "name": "たまねぎ"
            },
            {
                "name": "じゃがいも"
            },
            {
                "name": "牛肉"
            }
        ]
        },
        {
        "name": "炒飯",
        "category": "ご飯",
        "country": "中華",
        "season": "Any",
        "ingredients":[
            {
                "name": "人参"
            },
            {
                "name": "たまねぎ"
            },
            {
                "name": "卵"
            },
            {
                "name": "にんにく"
            }
        ]
        },
        {
        "name": "炊き込みご飯",
        "category": "ご飯",
        "country": "和食",
        "season": "Any",
        "ingredients":[
            {
                "name": "人参"
            },
            {
                "name": "椎茸"
            },
            {
                "name": "鶏肉"
            },
            {
                "name": "大葉"
            }
        ]
        }
]

JSONデータの取り込み

  1. //JSONファイルのパスを取得
  2. guard let path = Bundle.main.path(forResource: "RecipesList", ofType: "json") else { return }
  3. let url = URL(fileURLWithPath: path)
  4. do {
  5. let realm = try! Realm()
  6.  
  7. let data = try Data(contentsOf: url)
  8. let recipe_obj = try! JSONDecoder().decode([Recipe].self, from: data)
  9.  
  10. try! realm.write {
  11. realm.deleteAll()
  12. realm.add(recipe_obj, update: true)
  13. //人参を使うレシピを取得・表示
  14. var objects: Results<Recipe>
  15. objects = realm.objects(Recipe.self).filter("SUBQUERY(ingredients, $ingredient, $ingredient.name = %@).@count >= 1","人参")
  16. print("objectsの中",objects)
  17. }
  18. } catch {
  19. print("エラー")
  20. }

参考にしたサイト

https://qiita.com/cottpan/items/4bb5ad8bce2e28c0220a

0 件のコメント:

コメントを投稿