如何从协程范围返回值

时间:2020-07-18 07:36:01

标签: android kotlin kotlin-coroutines

有没有办法从协程范围返回值?例如这样的

suspend fun signUpUser(signUpData : SignUpUserRequest) : User {

CoroutineScope(Dispatchers.IO).launch() {

        val response = retrofitInstance.socialSignUP(signUpData)
        if (response.success) {
            obj = response.data
        } else {
                obj = response.message
        }
    }.invokeOnCompletion {
        obj = response.message
    }

    return obj
}

问题是signupUser函数被调用,返回obj暂停语句立即运行,而不是带有响应值。它没有按顺序运行,我期望的是首先响应,然后函数返回obj,但是没有发生。为什么?

我尝试了运行阻止,它达到了目的,但是有没有更好的方法来执行相同的任务而没有运行阻止?还是可以吗?

谢谢!

3 个答案:

答案 0 :(得分:4)

如果需要并行运行作品,则可以使用coroutineScopeasync协程生成器函数。但是,在您的情况下,您的请求不多,因此可能是 sequantial 。如上一个答案所述,您可以使用withContext。这是docs的一些小解释:

使用给定的协程上下文调用指定的暂停块, 暂停直到完成,然后返回结果。

suspend fun getUser(signUpData: SignUpUserRequest): User = withContext(Dispatchers.IO) {
    // do your network request logic here and return the result
}

请注意,当您使用带有函数的表达式主体时,请始终尝试显式注释返回类型。

编辑

协程使用常规的Kotlin语法来处理异常:try/catch或内置的辅助功能,例如runCatching(内部使用 try / catch )。 请注意,将始终抛出未捕获的异常。但是,不同的协程构建器以不同的方式对待异常。阅读official documentation或其他来源(例如this)总是很好。

答案 1 :(得分:3)

您可以使用withContext函数在后台线程中运行代码并返回一些结果:

suspend fun signUpUser(signUpData : SignUpUserRequest): User = withContext(Dispatchers.IO) {    
    val response = retrofitInstance.socialSignUP(signUpData)
    if (response.success) response.data else response.message
}

答案 2 :(得分:0)

在 Google 代码实验室 "Load and display images from the Internet" 中,有一个很好且非常优雅的 MutableLiveData 示例。

大纲:

  • 您使用 LiveData(内部)和 MutableLiveData(外部)创建一个 ViewModel
  • 现在您可以直接在视图、片段或活动中使用数据

好处是这些数据是生命周期感知的,您可以使用观察者的返回值。

class OverviewViewModel : ViewModel() {

    enum class MarsApiStatus { LOADING, ERROR, DONE }

    private val _status = MutableLiveData<MarsApiStatus>()
    val status: LiveData<MarsApiStatus> = _status

    private val _photos = MutableLiveData<List<MarsPhoto>>()
    val photos: LiveData<List<MarsPhoto>> = _photos

    /**
     * Call getMarsPhotos() on init so we can display status immediately.
     */
    init {
        getMarsPhotos()
    }

    /**
     * Gets Mars photos information from the Mars API Retrofit service and updates the
     * [MarsPhoto] [List] [LiveData].
     */
    private fun getMarsPhotos() {
        viewModelScope.launch {
            _status.value = MarsApiStatus.LOADING
            Log.d(TAG, "loading")
            try {
                _photos.value = MarsApi.retrofitService.getPhotos()
                _status.value = MarsApiStatus.DONE
                Log.d(TAG, "done")
            } catch (e: Exception) {
                _status.value = MarsApiStatus.ERROR
                _photos.value = listOf()
                Log.d(TAG, "error")
            }
        }
    }
}

您可以通过观察或直接在这样的视图中使用状态或照片值来使用它们:

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/photos_grid"
        android:layout_width="0dp"
        android:layout_height="0dp"
        app:layoutManager="androidx.recyclerview.widget.GridLayoutManager"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:listData="@{viewModel.photos}"
        app:spanCount="2"
        tools:itemCount="16"
        tools:listitem="@layout/grid_view_item" />

...在您“注册”并连接 f.ex 中的所有内容之后。关于片段,像这样:

override fun onCreateView(
    inflater: LayoutInflater, container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {

    try {
        val binding = FragmentOverviewBinding.inflate(inflater)
        //val binding = GridViewItemBinding.inflate(inflater)

        // Allows Data Binding to Observe LiveData with the lifecycle of this Fragment
        binding.lifecycleOwner = this

        // Giving the binding access to the OverviewViewModel
        binding.viewModel = viewModel

        binding.photosGrid.adapter = PhotoGridAdapter()

        return binding.root
    } catch (e: Exception) {
        Log.e("OverviewFragment", "${e.localizedMessage}")
    }
    return null
}