Posso osservare LiveData da più frammenti. Posso farlo con Flow? Se sì allora come?
Sì. Puoi farlo con emit
e collect
. Pensa emit
è simile ai dati live postValue
ed collect
è simile a observe
. Facciamo un esempio.
deposito
// 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()
}
Frammento
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
}
}
}
Possiamo avere più LiveData da un singolo LiveData usando map & switchMap. Esiste un modo per avere più flussi da un singolo flusso sorgente?
Flow è molto utile. Puoi semplicemente creare un flusso all'interno del flusso. Diciamo che si desidera aggiungere il segno di laurea a ciascuno dei dati delle previsioni del tempo.
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)
}
}
}
Quindi raccogliere i dati nel frammento come il n. 1. Qui ciò che accade è che il modello di visualizzazione sta raccogliendo dati dal repository e il frammento sta raccogliendo dati dal modello di visualizzazione.
Usando MutableLiveData posso aggiornare i dati da qualsiasi luogo usando il riferimento variabile. Esiste un modo per fare lo stesso con Flow?
Non è possibile emettere valore al di fuori del flusso. Il blocco di codice all'interno del flusso viene eseguito solo quando è presente un raccoglitore. Ma puoi convertire il flusso in dati live utilizzando l'estensione asLiveData da LiveData.
ViewModel
fun getWeatherForecast(): LiveData<String> {
return forecastRepository
.getWeatherForecastEveryTwoSeconds()
.asLiveData() // Convert flow to live data
}
Nel tuo caso puoi farlo
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
}
modificare
Grazie a @mark per il suo commento. La creazione di un nuovo flusso nel modello di visualizzazione per la getWeatherForecast
funzione non è effettivamente necessaria. Potrebbe essere riscritto come
fun getWeatherForecast(): Flow<String> {
return forecastRepository
.getWeatherForecastEveryTwoSeconds(spendingDetailsRequest)
.map {
it + " °C"
}
}