Я можу спостерігати LiveData з декількох фрагментів. Чи можу я це зробити за допомогою Flow? Якщо так, то як?
Так. Це можна зробити за допомогою emit
і collect
. Думаю 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
}
}
}
Ми можемо мати декілька LiveData з однієї LiveData, використовуючи map & switchMap. Чи є спосіб мати декілька потоків з одного джерела потоку?
Потік дуже зручний. Ви можете просто створити потік всередині потоку. Скажімо, ви хочете додати знак ступеня до кожного з даних прогнозу погоди.
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. Тут відбувається те, що модель перегляду збирає дані з сховища, а фрагмент збирає дані з перегляду.
Використовуючи MutableLiveData, я можу оновлювати дані з будь-якого місця за допомогою змінної посилання. Чи можна зробити те ж саме з Flow?
Ви не можете виділяти значення поза потоком. Блок коду всередині потоку виконується лише тоді, коли є будь-який колектор. Але ви можете конвертувати потік у живі дані, використовуючи розширення asLiveData від LiveData.
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"
}
}