🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Firebase

Firebaseは、Googleが提供するBasSサービスの一つ。リアルタイム通知可能、並びにアクセス制御ができるオブジェクトデータベース機能を備えます。さらに認証機能、アプリケーションのログ解析機能などの利用も可能です。

iOS

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

Xcode

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

Swift

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

Q&A

解決済

2回答

3413閲覧

【swift】同期処理が上手くいかずtableviewが上手く表示できません【firebase】

RyotaroIsoyama

総合スコア183

Firebase

Firebaseは、Googleが提供するBasSサービスの一つ。リアルタイム通知可能、並びにアクセス制御ができるオブジェクトデータベース機能を備えます。さらに認証機能、アプリケーションのログ解析機能などの利用も可能です。

iOS

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

Xcode

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

Swift

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

0グッド

1クリップ

投稿2019/12/12 19:07

編集2019/12/16 10:17

前提・実現したいこと

現在、xcode(swift)とfirestoreを用いて、DBに入っているフォロワー情報から、フォロワー一覧をtableviewに表示しようと試みています。

しかし同期処理やカスタムセルについて理解不足のためか上手くいきません。ご教授くださいませ。

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

Thread 1: Fatal error: Index out of range

該当のソースコード

swift

1// セルを返す 2 func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 3 let cell = tableView.dequeueReusableCell(withIdentifier: "SocialTableViewCell", for: indexPath) as! SocialTableViewCell 4 DispatchQueue.global().async { 5 self.fetchSocialTable() 6 DispatchQueue.main.async { 7 let data = self.socialTable[indexPath.row] 8 cell.UserName.text = data.UserName 9 cell.UserID.text = data.UserID 10 } 11 } 12 return cell 13 } 14 15 func fetchSocialTable(){ 16 self.db.collection("users").document(self.uid).collection("followers").limit(to:20).getDocuments(){snapshot,err in 17 if let snapshot = snapshot{ 18 self.socialTable = snapshot.documents.map{ userData -> SocialTable in 19 let data = userData.data() 20 return SocialTable(iconname: "hogehoge", UserName: data["UserName"] as! String, UserID: data["UserID"] as! String) 21 } 22 } 23 } 24 }

該当部分のソースコードを抜粋しましたが、

let data = self.socialTable[indexPath.row]

この部分で上記エラーが生じます。
printで確認したところ、socialTableには何も入っていませんでした。
socialTableは以下で初期化しています。

struct SocialTable{ var iconname: String var UserName: String var UserID: String } private var socialTable: [SocialTable] = []

fetchSocialTable関数内にあるself.uidにドキュメント名が正しく入っていることは確認済みです。
print関数で確認したところ、fetchSocialTable関数の通信処理が始まる前に該当のエラー箇所が実行されているようです。
同期処理は正しく書けていると思っているのですが、やはり間違っているのでしょうか。

ご教授いただきたいです。また、何か情報が足りなければお申し付けください。

追記

swift

1// セルを返す 2 func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 3 let cell = tableView.dequeueReusableCell(withIdentifier: "SocialTableViewCell", for: indexPath) as! SocialTableViewCell 4 let queue = DispatchQueue.global(qos: .default) 5 let dataset = self.fetchSocialTable() 6 queue.async { 7 cell.UserName.text = dataset[indexPath.row].UserName 8 cell.UserID.text = dataset[indexPath.row].UserID 9 } 10 11 return cell 12 } 13 14func fetchSocialTable() -> [SocialTable]{ 15 let semaphore = DispatchSemaphore(value: 0) 16 var data:[SocialTable] = [] 17 self.db.collection("users").document(self.uid).collection("followers").limit(to:20).getDocuments(){snapshot,err in 18 if let snapshot = snapshot{ 19 print("connect firestore...") 20 for document in snapshot.documents { 21 data.append(SocialTable(iconname: "hogehoge", UserName: document.data()["Nm"] as! String, UserID: document.data()["UI"] as! String)) 22 } 23 } 24 semaphore.signal() 25 print("semaphore ok") 26 print(self.socialTable) 27 } 28 print("semaphore wait...") 29 semaphore.wait() 30 print("complete...") 31 return data 32 }

これで実行した場合、semaphore wait...のみが出力されその後の処理が走りません。
一旦テーブルへの書き込みをやめて、firestoreからデータを取得する部分の確認を行ったのですが、それは上手くいっていました。
何かいけない点があるのでしょうか。

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

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

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

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

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

fuzzball

2019/12/18 05:37

tableView(_:numberOfRowsInSection:)を見せて下さい。
RyotaroIsoyama

2019/12/18 13:53

// セルの数 func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { if(followOrFollower_count>20){ return 20 }else{ return followOrFollower_count } }
RyotaroIsoyama

2019/12/18 13:55

見にくくてすみません、 followOrFollower_countは全画面から遷移する際に値を入れています! 今回は2が入っています
guest

回答2

0

解決しました。semaphore.wait()でデッドロックが起こっていたのが問題だったみたいです。

swift

1let queue = DispatchQueue.global() 2let mainQueue = DispatchQueue.main 3queue.async { 4 let dataset = self.fetchSocialTable() 5 mainQueue.async { 6 cell.UserName.text = dataset[indexPath.row].UserName 7 cell.UserID.text = dataset[indexPath.row].UserID 8 } 9}

投稿2019/12/18 19:18

RyotaroIsoyama

総合スコア183

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

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

0

ベストアンサー

データの読み込みが終わってないのに、

swift

1let data = self.socialTable[indexPath.row]

を呼び出したためだと思います。
メインスレッドに戻る前に、データの読み込みが終わったかどう確認してみてください。

GCPのDispatchSemaphoreを使えば、上手く行けるはずです。
以下は、APIコールを待っている処理の事例です。

import Foundation class MyTest { func runTheShow() { DispatchQueue.global(qos: .utility).async { let json: Teratail? = self.fetchQuestion() DispatchQueue.main.async { print(json) } } } func fetchQuestion() -> Teratail? { var myresponse: Teratail? = nil guard let url = URL(string: "https://teratail.com/api/v1/questions") else {return nil} let semaphore = DispatchSemaphore(value: 0) let task = URLSession.shared.dataTask(with: url) { (data, response, error) in do { guard let data = data else {return} let json = try JSONDecoder().decode(Teratail.self, from: data) myresponse = json } catch { print("error:", error) } semaphore.signal() } task.resume() _ = semaphore.wait(wallTimeout: .distantFuture) return myresponse } } struct Teratail: Decodable { let meta: Meta let questions: [Question] } struct Meta: Decodable { let hit_num: Int let limit: Int let message: String let page: Int let total_page: Int } struct Question : Decodable { let count_clip: Int let count_pv : Int let count_reply: Int let created: String let modified: String let id: Int let is_accepted: Bool let is_beginner: Bool let is_presentation: Bool let tags: [String] let title: String let user: User } struct User: Decodable { let display_name: String let photo: String let score: Int } let obj = MyTest() obj.runTheShow()

投稿2019/12/13 00:47

編集2019/12/13 01:47
vanderlvov

総合スコア687

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

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

RyotaroIsoyama

2019/12/16 10:12 編集

アドバイスありがとうございます!!!! semaphoreを使ってみたのですが、今度はsemaphore.wait()で処理が止まるようです。 コードが読みづらいので本文に追記することにします。
vanderlvov

2019/12/17 11:24

「semaphore wait...」で止まると言うのは、「semaphore.signal()」まで行けないようですね。 この辺のログをみてもいいかも: if let snapshot = snapshot{ print("connect firestore...") for document in snapshot.documents { data.append(SocialTable(iconname: "hogehoge", UserName: document.data()["Nm"] as! String, UserID: document.data()["UI"] as! String)) } }
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

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

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

質問する

関連した質問