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

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

ただいまの
回答率

90.51%

  • Android

    6522questions

    Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

  • Firebase

    605questions

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

  • Kotlin

    326questions

    Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。

Firebase Realtime Databaseでのデータ取得について【Android】

解決済

回答 1

投稿 編集

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

skliber

score 23

 やりたいこと

Firebaseにある特定のデータを取得してRecyclerViewに表示したいです。
具体的には、Realtime Database上でFirebaseAuthのuidを持つuserを探し、そのuserが持っているfriendsの一覧を表示したいです。

 問題と質問

できるにはできたのですが、あまりにも無理矢理な感が強く、とても正しい方法だと思えません。
もっと良い方法を教えて頂きたいと思っています。

 現在のコード

  • Firebase Realtime Database
{
  "users" : {
    "userkey01" : { // push()すると作られる固有のキーです
      "friends" : {
        "userkey02" : "user2name" // このユーザーのフレンドがfirebase上のキーとnameの値で格納されています
      },
      "name" : "user1",
      "uid" : "user1uid" // FirebaseAuthのUIDです
    },
    "userkey02" : {
      "friends" : {
        "userkey01" : "user1name"
      },
      "name" : "user2",
      "uid" : "user2uid"
    }
  }
}
  • Fragmentのコード抜粋
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        val view = inflater!!.inflate(R.layout.fragment_friend, container, false)

        val recyclerView = view.friend_recycler
        recyclerView.layoutManager = LinearLayoutManager(activity)

        val array = ArrayList<String>() // このArrayListにfriendのnameが入ります

        val databaseRef: DatabaseReference = FirebaseDatabase.getInstance().reference
        val currentUserUid = FirebaseAuth.getInstance().currentUser!!.uid

        // 現在ログインしているユーザーのuidから、それを持つノードを検索しています
        val currentUserRef = databaseRef.child(FirebaseEntityName.USER).orderByChild("uid").equalTo(currentUserUid)

     // 検索したユーザーのノードを一度だけ取得します
        currentUserRef.addListenerForSingleValueEvent(object : ValueEventListener {
            override fun onDataChange(p0: DataSnapshot?) {

                // 見つかっていればfirstChildにノードが入るはずなので、格納します
                val firstChild = p0!!.children.iterator().next()

                // 見つかったuserのkeyを使用して、userの持つfriendsの参照を取得します
                var friends = databaseRef.child(FirebaseEntityName.USER).child(firstChild.key).child("friends")

                // friendsのデータを取得します
                friends.addListenerForSingleValueEvent(object : ValueEventListener {

                    override fun onDataChange(p0: DataSnapshot?) {

                        // 見つかっていれば格納します
                        // 現在データが一つしかありませんが、複数の予定なので、iteratorを使ってループで回すことになると思います。
                        val firstChild = p0!!.children.iterator().next()

                        // ArrayListに名前を追加して、notifyItemChanged()でrecyclerViewに変更を通知します
                        array.add(firstChild.value)
                        recyclerView.adapter.notifyItemChanged(array.size - 1)
                    }

                    override fun onCancelled(p0: DatabaseError?) {
                        Log.d("cancelled", "cancelled")
                    }
                })
            }

            override fun onCancelled(p0: DatabaseError?) {
                Log.d("cancelled", "cancelled")
            }
        })

        val itemDecoration = DividerItemDecoration(activity, DividerItemDecoration.VERTICAL)
        recyclerView.addItemDecoration(itemDecoration)

        recyclerView.adapter = FriendAdapter(activity,array)
        return view
    }

よろしくお願いします。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

check解決した方法

0

 1

onDataChange()をネストさせて複雑なクエリを作るというのは、それほど的を外した考えではなかったようです

 2

ただ、今回の例ではval firstChild = p0!!.children.iterator().next()の時点でデータが取得出来ているので、複雑に考えずこのvalueを使用すれば済みました
このときgetValue()を使用してオブジェクトとマッピングするのが常套手段であるようですが、私のモデルのせいか、このメソッドは例外が出て動きませんでした

  • モデル
data class User(
        var uid: String,
        var name: String,
        var friends: Map<String, String>
)
  • 例外コード
currentUserRef.addListenerForSingleValueEvent(object : ValueEventListener {
    override fun onDataChange(p0: DataSnapshot?) {
        val firstChild = p0!!.children.iterator().next()

        // 以下、どちらも com.google.firebase.database.DatabaseException: Class com.example.model.entity.User is missing a constructor with no arguments
        val value: User? = p0!!.getValue(User::class.java)
        val value: User? = firstChild.getValue(User::class.java)
    }
}

 3

いろいろ試したのですがどうにもこの例外が解決できないので、
getValue()ではなくMoshiというライブラリを使ってパースしたところ、うまく行きました

  • 解決
currentUserRef.addListenerForSingleValueEvent(object : ValueEventListener {
    override fun onDataChange(p0: DataSnapshot?) {

        val firstChild = p0!!.children.iterator().next()
        val moshi = Moshi.Builder().build()
        val adapter = moshi.adapter(User::class.java)
        val json = adapter.fromJsonValue(firstChild.value)

        Log.d("json", json.toString())
        val user = json as User

        Log.d("uid", user.uid)
        Log.d("name", user.name)
        Log.d("friends", user.friends.toString())

        user.friends.forEach {
            array.add(it.value)
            recyclerView.adapter.notifyItemChanged(array.size - 1)
        }

    }

    override fun onCancelled(p0: DatabaseError?) {
        Log.d("cancelled", "cancelled")
    }
})

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

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

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

関連した質問

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

  • Android

    6522questions

    Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

  • Firebase

    605questions

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

  • Kotlin

    326questions

    Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。