Kotlin FlowとAndroid LiveData


20

Kotlin Flowについて質問があります

  1. 複数のフラグメントからのLiveDataを観察できます。これをFlowで実行できますか?はいの場合はどうですか?
  2. map&を使用して、1つのLiveDataから複数のLiveDataを取得できますswitchMap。単一のソースフローから複数のフローを作成する方法はありますか?
  3. 使用してMutableLiveData、私は、変数の参照を使用してどこからでもデータを更新することができます。Flowで同じことをする方法はありますか?

私は次のようなユースケースを持っています。単一のソースフローを提供するSharedPreferences使用法callbackFlow{...}を観察します。そのフローから、キーと値のペアごとに複数のフローを作成します。

これらはばかげた質問に聞こえるかもしれません。RxとFlowの世界は初めてです。


FlowLiveDataのどちらのアプローチを採用しましたか?
IgorGanapolsky

2
現在、ビューにはLiveDataを使用し、その他すべてにはFlowを使用しています。ViewModelでは、Flowを受け取り、LiveDataを発行してフラグメントから観察します。
zoha131

@ zoha131あなたは正しい方法でそれを行います!LiveDataはメインスレッドでのみ監視できるため、View <-> ViewModelの相互作用に完全に適合します。次に、フローを使用すると、アーキテクチャの残りの部分でより複雑な操作を行うことができます。
smora

回答:


15

複数のフラグメントからのLiveDataを観察できます。これをFlowで実行できますか?はいの場合はどうですか?

はい。これはemitおよびで行うことができますcollect。Think emitはライブデータに似てpostValueおり、collectに似ていobserveます。例を挙げましょう。

リポジトリー

// I just faked the weather forecast
val weatherForecast = listOf("10", "12", "9")

// This function returns flow of forecast data
// Whenever the data is fetched, it is emitted so that
// collector can collect (if there is any)
fun getWeatherForecastEveryTwoSeconds(): Flow<String> = flow { 
    for (i in weatherForecast) {
        delay(2000)
        emit(i)
    }
}

ViewModel

fun getWeatherForecast(): Flow<String> {
    return forecastRepository.getWeatherForecastEveryTwoSeconds()
}

断片

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
    // Collect is suspend function. So you have to call it from a 
    // coroutine scope. You can create a new coroutine or just use 
    // lifecycleScope
    // https://developer.android.com/topic/libraries/architecture/coroutines
    lifecycleScope.launch {
            viewModel.getWeatherForecastEveryTwoSeconds().collect {
                    // Use the weather forecast data
                    // This will be called 3 times since we have 3 
                    // weather forecast data
            }
    }
}

map&switchMapを使用して、単一のLiveDataから複数のLiveDataを作成できます。単一のソースフローから複数のフローを作成する方法はありますか?

流れがとても便利です。フロー内にフローを作成できます。各天気予報データに度合記号を追加するとします。

ViewModel

fun getWeatherForecast(): Flow<String> {
    return flow {
        forecastRepository
            .getWeatherForecastEveryTwoSeconds(spendingDetailsRequest)
                .map {
                    it + " °C"
                }
                .collect {
                    // This will send "10 °C", "12 °C" and "9 °C" respectively
                    emit(it) 
                }
    }
}

次に、1と同じFragmentでデータを収集します。ここでは、ビューモデルがリポジトリからデータを収集し、フラグメントがビューモデルからデータを収集しています。

MutableLiveDataを使用すると、変数参照を使用してどこからでもデータを更新できます。Flowで同じことをする方法はありますか?

フローの外に値を放出することはできません。フロー内のコードブロックは、コレクターがある場合にのみ実行されます。ただし、LiveDataのasLiveData拡張機能を使用して、フローをライブデータに変換できます。

ViewModel

fun getWeatherForecast(): LiveData<String> {
    return forecastRepository
    .getWeatherForecastEveryTwoSeconds()
    .asLiveData() // Convert flow to live data
}

あなたの場合、これを行うことができます

private fun getSharedPrefFlow() = callbackFlow {
    val sharedPref = context?.getSharedPreferences("SHARED_PREF_NAME", MODE_PRIVATE)
    sharedPref?.all?.forEach {
        offer(it)
    }
}

getSharedPrefFlow().collect {
    val key = it.key
    val value = it.value
}

編集する

彼のコメントを@markに感謝します。getWeatherForecast関数のビューモデルで新しいフローを作成することは実際には不要です。次のように書き直すことができます

fun getWeatherForecast(): Flow<String> {
        return forecastRepository
                .getWeatherForecastEveryTwoSeconds(spendingDetailsRequest)
                    .map {
                        it + " °C"
                    }
    }

理由はわかりませんが、1つのFlowに対して複数の場所でcollect()関数を呼び出すことができないという想定がありました。答えてくれてありがとう。
zoha131

1
いいえ。同じフローを複数の場所で収集できます。val sharedPref = getSharedPref()複数の場所で収集を使用できますsharedPref.collect {}。唯一のことは、コレクトがサスペンドであるため、コルーチンブロックから呼び出す必要があることです。そしてnpを助けてうれしい:)
Fatih

私の3番目の質問については、回避策は放送チャンネルかもしれません。
zoha131

ライブデータの代わりにチャネルを使用するために、このコミットを確認できます。github.com/android/plaid/pull/770/commits/...
ファティ

1
はい、そうです。これが流れの出所です。チャネルには、注意する必要がある多くのものがあり、観測者がいない場合でも常に開いているという意味でホットです。しかし、流れがあれば、寒いので心配することなく同じ利点を得ることができます。したがって、チャネルの代わりに、フローを使用する方が良いと思います
Fatih

3

Flow.asLiveData()新しいandroidx.lifecyclektxパッケージには、新しい拡張機能があります。あなたは私の記事でもっと学ぶことができます:https//www.netguru.com/codestories/android-coroutines-%EF%B8%8Fin-2020


これはいつ使用する必要がありますか?
IgorGanapolsky

1
Flowインスタンスを持つLiveDataを必要とするAPIを満足させたい場合
Samuel Urbanowicz

Googleによると、我々がしなければならないのいずれかを選択します LiveData またはフロー:codelabs.developers.google.com/codelabs/...
IgorGanapolsky

1

3層アーキテクチャでは、データドメインプレゼンテーション、フローはデータレイヤー(データベース、ネットワーク、キャッシュ...)で行われる必要があります。次に、サミュエルアーバノビッチが述べたように、フローをLiveDataにマッピングできます。

一般に、FlowはRxJavaのObservable(またはFlowable)とほぼ同じです。LiveDataと混同しないでください。

詳細:https : //medium.com/@elizarov/cold-flows-hot-channels-d74769805f9


ワンショット操作(データベースの読み取りなど)には、LiveDataで十分です。
IgorGanapolsky
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.