Retroift coroutines livedata 仅在第一次调用

Umu*_*kin 4 android kotlin android-livedata android-architecture-components kotlin-coroutines

我只是 kotlin 协程的新手。我刚刚创建了用于测试实时数据的新项目,但我无法观察到数据更改。我不明白 livedata 的概念。什么时候触发?因为当我观察 ROOM 数据库时(不是协程方式。我使用了 MutableLiveData)它工作得很好。每当数据发生变化时,总是会触发观察者。

我只是想清理和现代代码。我的期望:当我单击 btnLogin 按钮时(当用户使用另一个帐户登录时或者您可以说数据更改时)必须触发 livedata。

这是我的例子:

改造界面:

interface RetroMainClient {

    @POST("login.php")
    suspend fun login(@Body model: UserLoginModel): Response<UserLoginModel>

    companion object {
        val getApi: RetroMainClient by lazy {
            Retrofit.Builder().baseUrl("https://example.com/")
                .addConverterFactory(GsonConverterFactory.create(GsonBuilder().create())).build()
                .create(RetroMainClient::class.java)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我的存储库:

class Repository {
    suspend fun getLoginApi(model: UserLoginModel) = RetroMainClient.getApi.login(model)
}
Run Code Online (Sandbox Code Playgroud)

我的视图模型:

class MainViewModel : ViewModel() {

    fun login(model: UserLoginModel) = liveData(IO) {
        try {
            emit(Repository().getLoginApi(model))
        } catch (e: Exception) {
            Log.e("exception", "${e.message}")
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

和我的主要活动:

class MainActivity : AppCompatActivity() {

    private lateinit var viewModel: MainViewModel

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

        var model = UserLoginModel("user1", "123456")

        viewModel.login(model).observe(this, Observer {

            if (it.isSuccessful) {
                btnLogin.text = it.body()?.username
            }

        })

        btnLogin.setOnClickListener {
            model = UserLoginModel("user2", "123456")

            CoroutineScope(IO).launch {
                try {
                    Repository().getLoginApi(model)
                } catch (e: Exception) {
                    Log.e("exception:", "${e.message}")
                }
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

Ser*_*gey 6

当您调用viewModel.login()方法时,您创建了一个新的LiveData类实例。为了viewModel.login()在每次单击btnLogin按钮后执行相应的块,您需要LiveData.observe()为每次viewModel.login()调用调用方法。

InMainActivityonCreate方法:

btnLogin.setOnClickListener {
    model = UserLoginModel("user2", "123456")
    viewModel.login(model).observe(this, Observer { data ->
        if (it.isSuccessful) {
            btnLogin.text = data.body()?.username
        }
    })
}
Run Code Online (Sandbox Code Playgroud)

另一种方法

是在类中启动协程MainViewModelLiveData手动更新字段:

class MainViewModel : ViewModel() {

    val loginResponse: LiveData<Response<UserLoginModel>> = MutableLiveData<Response<UserLoginModel>>()

    fun login(model: UserLoginModel) = viewModelScope.launch(IO) {
        try {
            (loginResponse as MutableLiveData).postValue(Repository().getLoginApi(model))
        } catch (e: Exception) {
            Log.e("exception", "${e.message}")
        }
    }
}


class MainActivity : AppCompatActivity() {

    private lateinit var viewModel: MainViewModel

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        viewModel = ViewModelProvider(this).get(MainViewModel::class.java)

        var model = UserLoginModel("user1", "123456")

        viewModel.loginResponse.observe(this, Observer {

            if (it.isSuccessful) {
                btnLogin.text = it.body()?.username
            }

        })

        btnLogin.setOnClickListener {
            model = UserLoginModel("user2", "123456")

            viewModel.login(model)
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

viewModelScopeMainViewModel类中使用,向 build.gradle 文件添加依赖项:

final LIFECYCLE_VERSION = "2.2.0-rc03" // 添加最新版本

api "androidx.lifecycle:lifecycle-viewmodel-ktx:$LIFECYCLE_VERSION"

  • 我不知道该如何感谢你。我研究了几个小时,我知道每次单击按钮时我都会创建新实例。但我不知道如何避免它。太感谢了。 (2认同)