我在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
}