Livedata 观察者多次触发

Mic*_*ell 5 android mvvm android-fragments kotlin android-livedata

我在用我的项目设置 MVVM 时遇到了麻烦,一切似乎都在顺利运行,除非我注意到我的 LiveData 对象的观察者被多次调用。有时为同一个 LiveData 对象更新一次 4-5 次。

主活动上托管了多个片段,所有片段都链接到连接到我的存储库的同一个 ViewModel,然后使用改造向我的服务器发出 API 请求。

然后我在每个片段中都有单独的观察者,观察来自 ViewModel 的相同 LiveData 对象的变化,当使用单个片段时,我没有对同一变化进行多次观察的问题,但是在将项目扩展为具有多个片段时我注意到观察者被称为很多。

我已将观察者更改为全部使用 viewLifecycleOwner 无济于事。

这是一个片段示例,每个片段在观察 LiveData 的方式上基本相同,只是它们更新 UI 的不同部分。

override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
        viewModel = ViewModelProviders.of(this, viewModelFactory)
                .get(WeatherViewModel::class.java)
        bindUI()
    }

    private fun bindUI() = launch {
        val currentWeatherResponse = viewModel.weatherResponse.await()

        currentWeatherResponse.observe(viewLifecycleOwner, Observer {
            if(it == null) return@Observer
            Log.i("CurrentWeatherFragment", "Observed")
            val currentWeather = it.currentWeather.first()
            updateTemperature(currentWeather.temperature, currentWeather.tempMin, currentWeather.tempMax)
            updateWind(currentWeather.windSpeed, currentWeather.windDirName)
            updateHumidity(currentWeather.humidity)

        })
    }
Run Code Online (Sandbox Code Playgroud)

这是每个片段连接到的 ViewModel:

class WeatherViewModel(
        private val forecastRepository: IForecastRepository
) : ViewModel() {

    val weatherResponse by lazyDeferred {
        forecastRepository.getWeatherResponse()
    }

}
Run Code Online (Sandbox Code Playgroud)

我的存储库中的天气响应方法:

    override suspend fun getWeatherResponse(): LiveData<out DetailedPrediction> {
        return withContext(Dispatchers.IO) {
            initWeatherData()
            println("Get Weather response method")
            println(currentWeatherDao.getWeatherResponse().value)
            return@withContext currentWeatherDao.getWeatherResponse()
        }
    }
Run Code Online (Sandbox Code Playgroud)

还有从 saveWeatherResponse 方法接收数据的天气 DAO:

    private val weatherResponse = MutableLiveData<DetailedPrediction>()
    private val radarImages = MutableLiveData<RadarImageList>()

    //TODO Return weather response from shared preferences
    override fun getWeatherResponse(): LiveData<DetailedPrediction> {
        return weatherResponse
    }
Run Code Online (Sandbox Code Playgroud)

我希望能够从这个单一的 LiveData 事件更新我的所有 UI 元素,这实际上是有效的,但观察者触发的次数太多了,我终生都无法弄清楚原因。

Mic*_*ell 7

我不敢相信我花了这么长时间才找出一个简单的错误,但是如果有人遇到这个问题(我对此表示怀疑),我会在每次观察中添加一个片段而不是替换该片段,这意味着一个全新的观察者是为每个片段创建。

这是旧代码:


    fun addDailyWeatherFragment() {
        val fragmentTransaction: FragmentTransaction = fragmentManager!!.beginTransaction()
        val dailyWeatherFragment = DailyWeatherFragment()
        fragmentTransaction.add(R.id.fragmentContainer, dailyWeatherFragment)
        fragmentTransaction.commit()
    }
Run Code Online (Sandbox Code Playgroud)

和固定代码:


    fun addDailyWeatherFragment() {
        val fragmentTransaction: FragmentTransaction = fragmentManager!!.beginTransaction()
        val dailyWeatherFragment = DailyWeatherFragment()
        fragmentTransaction.replace(R.id.fragmentContainer, dailyWeatherFragment)
        fragmentTransaction.commit()
    }
Run Code Online (Sandbox Code Playgroud)