我在Kotlin中为Android集成了一个简单的演示应用程序,该应用程序使用Jsoup检索了网页的标题。我正在使用Dispatchers.Main作为上下文进行网络 call 。

我对协程的理解是,如果我在launch上调用Dispatchers.Main,则它确实在主线程上运行,但会暂停执行以免阻塞线程。

我对android.os.NetworkOnMainThreadException的理解是存在的,因为网络操作很繁琐,并且在主线程上运行时会阻止它。

所以我的问题是,鉴于协程不会阻塞运行它的线程,所以NetworkOnMainThreadException真的有效吗?这是一些在Jsoup.connect(url).get()上抛出给定异常的示例代码:

class MainActivity : AppCompatActivity() {
    val job = Job()

    val mainScope = CoroutineScope(Dispatchers.Main + job)

    // called from onCreate()
    private fun printTitle() {
        mainScope.launch {
            val url ="https://kotlinlang.org"
            val document = Jsoup.connect(url).get()
            Log.d("MainActivity", document.title())
            // ... update UI with title
        }
    }
}

我知道我可以使用Dispatchers.IO上下文简单地运行它,并将结果提供给main / UI线程,但这似乎躲避了协程的某些实用程序。

作为参考,我正在使用Kotlin 1.3。

最佳答案



挂起执行以便不阻塞线程的唯一点是标记为suspend的方法-即挂起方法。

由于Jsoup.connect(url).get()不是挂起方法,因此它将阻塞当前线程。当您使用Dispatchers.Main时,当前线程是主线程,并且您的网络操作直接在主线程上运行,从而导致NetworkOnMainThreadException

get()方法这样的阻塞工作可以通过将其包装在withContext()中而挂起,一种挂起方法,并确保该方法运行时不会阻塞Dispatchers.Main

mainScope.launch {
    val url ="https://kotlinlang.org"
    val document = withContext(Dispatchers.IO) {
        Jsoup.connect(url).get()
    }
    Log.d("MainActivity", document.title())
    // ... update UI with title
}

07-26 04:00