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

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

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

JSON(JavaScript Object Notation)は軽量なデータ記述言語の1つである。構文はJavaScriptをベースとしていますが、JavaScriptに限定されたものではなく、様々なソフトウェアやプログラミング言語間におけるデータの受け渡しが行えるように設計されています。

Swift

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

API

APIはApplication Programming Interfaceの略です。APIはプログラムにリクエストされるサービスがどのように動作するかを、デベロッパーが定めたものです。

Q&A

解決済

1回答

1613閲覧

[Swift]楽天APIで取得したJSONデータを、Codableで任意の型に変換したい

kageusu

総合スコア2

JSON

JSON(JavaScript Object Notation)は軽量なデータ記述言語の1つである。構文はJavaScriptをベースとしていますが、JavaScriptに限定されたものではなく、様々なソフトウェアやプログラミング言語間におけるデータの受け渡しが行えるように設計されています。

Swift

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

API

APIはApplication Programming Interfaceの略です。APIはプログラムにリクエストされるサービスがどのように動作するかを、デベロッパーが定めたものです。

0グッド

0クリップ

投稿2021/07/21 09:47

前提・実現したいこと

Swiftで、好き嫌いから食材やレシピを提示するシステムを作っています。
楽天レシピのAPIからデータを取得し、Codableで任意のデータ型に変換しようとしたときに、
うまくデータを取得できない状況に陥りました。
structの書き方が原因なのかな?と思い、そこを重点的に書き換えていましたが、どうしてもうまくいかなかったため、質問を投稿しました。

###取得しているJSONデータ

Json

1{ 2 "result": { 3 "small": [ 4 { 5 "categoryName": "ソーセージ・ウインナー", 6 "parentCategoryId": "66", 7 "categoryId": 50, 8 "categoryUrl": "https://recipe.rakuten.co.jp/category/10-66-50/" 9 }, 10 ...(以下、この型のデータが数千)... 11 ], 12 "medium": [...], 13 "large": [...] 14 } 15}

コード全体

swift

1import UIKit 2 3//APIで取得したデータを使える形に直すために、Structでモデルを作ってあげる。 4struct Rakuten: Codable { 5 var result: Result 6 7 struct Result: Codable { 8 var small: [Food] 9 var medium: [Food] 10 var large: [Food] 11 12 struct Food: Codable { 13 var categoryName: String 14 var parentCategoryId: String 15 var categoryId: String 16 var categoryUrl: String 17 } 18 } 19 20} 21 22class ViewController: UIViewController { 23 24 override func viewDidLoad() { 25 super.viewDidLoad() 26 //APIデータを取得 27 getRakutenAPI() 28 } 29 30 private func getRakutenAPI() { 31 //取得できなかった時、処理を終了 32 guard let url = URL(string: "https://app.rakuten.co.jp/services/api/Recipe/CategoryList/20170426?format=json&applicationId=1022307395515225117") else { return } 33 34 var request = URLRequest(url: url) 35 request.httpMethod = "GET" 36 37 let task = URLSession.shared.dataTask(with: url) { (data, response, err) in 38 if let err = err { 39 print("情報の取得に失敗しました1:",err) 40 return 41 } 42 43 //上で書いたdataにデータがしっかり入っていたら走る処理 44 if let data = data { 45 do { 46/* 47 以下のコードでは、しっかりデータを取得できていることを確認 48 let json = try JSONSerialization.jsonObject(with: data, options: .fragmentsAllowed 49*/ 50 let rakuten = try JSONDecoder().decode([Rakuten].self, from: data) 51 print("rakuten: ", rakuten) 52 } catch(let err) { 53 print("情報の取得に失敗しました2:",err) 54 } 55 } 56 } 57 58 task.resume() 59 } 60}

該当のソースコード

swift

1do { 2/* 3 以下のコードでは、しっかりデータを取得できていることを確認 4 let json = try JSONSerialization.jsonObject(with: data, options: .fragmentsAllowed 5*/ 6 let rakuten = try JSONDecoder().decode([Rakuten].self, from: data) 7 print("rakuten: ", rakuten) 8 } catch(let err) { 9 print("情報の取得に失敗しました2:",err) 10 } 11}

発生している問題・エラーメッセージ

デバッグエリアに表示されるエラーです。
該当のソースコードの下から3行目
「print("情報の取得に失敗しました2:",err)」
が出力されてしまいます。

2021-07-15 17:15:37.413023+0900 practiceAPI[34761:5744376] [] nw_protocol_get_quic_image_block_invoke dlopen libquic failed 情報の取得に失敗しました2: typeMismatch(Swift.Array<Any>, Swift.DecodingError.Context(codingPath: [], debugDescription: "Expected to decode Array<Any> but found a dictionary instead.", underlyingError: nil))

試したこと

コード全体の冒頭にあるstructの書き方が間違っていると考え、様々な形に書き換えていましたが、全てデータを取得できない状況。

######書き換えたコード

swift

1struct Rakuten: Codable { 2 var result: Result 3} 4 5struct Result: Codable { 6 var small: [Food] 7 var medium: [Food] 8 var large: [Food] 9} 10 11struct Food: Codable { 12 var categoryName: String 13 var parentCategoryId: String 14 var categoryId: String 15 var categoryUrl: String 16}

[]でデータを囲んでみたり、さまざまな形で置き換えていました。
最終的に、全体のコードの形に落ち着きましたが、これ以上他の方法が思い浮かんでいない状況です。

参考にしたサイト

【Swift】Codableについて備忘録
【プログラミング】Swift:APIを使ってQiitaの記事を取得しよう!
APIをデバックエリアに表示させる〜Swiftで楽天レシピAPIを表示させてみた①〜

補足情報(FW/ツールのバージョンなど)

API:楽天レシピ API
Xcode 12.5.1
swift5

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

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

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

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

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

guest

回答1

0

ベストアンサー

JSON のトップレベルは Rakuten の配列ではなく Rakuten そのものなので、decode する型は [Rakuten].self ではなく Rakuten.self にすれば良いと思います。

diff

1- let rakuten = try JSONDecoder().decode([Rakuten].self, from: data) 2+ let rakuten = try JSONDecoder().decode(Rakuten.self, from: data)

また、"categoryId": 50, ってことは categoryId の型は String ではなく Int ですね。

投稿2021/07/21 10:39

hoshi-takanori

総合スコア7901

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

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

kageusu

2021/07/21 11:03 編集

解答いただきありがとうございます! hoshiさんの通りに修正しましたが、情報の取得に失敗してしまいました。 ですが、エラー文は変わりました 以下エラー文↓ 情報の取得に失敗しました: typeMismatch(Swift.String, Swift.DecodingError.Context(codingPath: [CodingKeys(stringValue: "result", intValue: nil), CodingKeys(stringValue: "small", intValue: nil), _JSONKey(stringValue: "Index 0", intValue: 0), CodingKeys(stringValue: "categoryId", intValue: nil)], debugDescription: "Expected to decode String but found a number instead.", underlyingError: nil)) 自分のstructの書き方が悪いのでしょうか? *すみません、StringからIntにまだ修正していなかったので、修正後またコメントします
kageusu

2021/07/21 11:27

ありがとうございます!! structの型を以下に修正しました struct Rakuten: Codable { var result: Result struct Result: Codable { var small: [Food] var medium: [Food] var large: [LargeCate] struct Food: Codable { //実際の型(IntとかString)も揃えないといけない var categoryName: String var parentCategoryId: String var categoryId: Int var categoryUrl: String } struct LargeCate: Codable { var categoryName: String var categoryId: String var categoryUrl: String } } } 結果、うまくいきました!! JSONの型と必ず一致させることと、自分のsmall,midium,largeの中身の確認不足であったことがわかりました。 ご教授いただきありがとうございました!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問