使用协程对任务进行排队

Azh*_*r92 8 android kotlin kotlin-coroutines

我最近开始阅读有关协程的内容,我想询问某个场景。

考虑一个带有一个按钮的简单屏幕。单击后,它会执行一堆打印语句和一些延迟(其间)。到目前为止,我正在使用协程来实现这一目标。

现在,我的问题是,如果用户反复向该按钮发送垃圾邮件,是否有一种方法可以将任务排队并在专用线程上逐个执行(一旦前一个任务执行完毕)?

override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
        val thread = newSingleThreadContext("thread")

        binding.button1.setOnClickListener {

            CoroutineScope(thread).launch {
               Log.d("ABC", " TEXT1 ")
               delay(1000)
               Log.d("ABC", " TEXT2 ")
               delay(2000)
            }

        }

}
Run Code Online (Sandbox Code Playgroud)

现在,如果我执行此操作,它会正常运行,并且如果我按一次按钮,我就可以看到日志:

TEXT1
TEXT2
Run Code Online (Sandbox Code Playgroud)

但是,如果我向按钮发送垃圾邮件,我会看到以下内容:

TEXT1
TEXT1
TEXT2
TEXT1
TEXT2
TEXT2
Run Code Online (Sandbox Code Playgroud)

基本上,根据我的观察,发生的情况是,当用户按下按钮时,协程就会在没有任何等待或检查的情况下执行。所以,我的问题是,如何编辑代码,以便当用户向按钮发送垃圾邮件时,每个 courotine 都会按顺序执行,并且每个 courotine 在下一个开始执行之前完成,以便输出如下所示:

TEXT1
TEXT2
TEXT1
TEXT2
TEXT1
TEXT2
Run Code Online (Sandbox Code Playgroud)

Gle*_*val 2

无论您使用什么线程或线程池,您都可以通过 Flow 获得该行为:

class MyActivity : CoroutineScope by MainScope() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        binding = DataBindingUtil.setContentView(this, R.layout.activity_main)

        val myFlow = callbackFlow {
            binding.button1.setOnClickListener {
                offer(Unit)
            }
            awaitClose()
        }

        launch(Dispatchers.Default) {
            myFlow
                .collect {
                    Log.d("ABC", " TEXT1")
                    delay(1000)
                    Log.d("ABC", " TEXT2")
                    delay(2000)
                }
        }

    }

    override fun onDestroy() {
        super.onDestroy()
        cancel()
    }
}
Run Code Online (Sandbox Code Playgroud)