android-NetworkOnMainThreadException对协程中的网络调用有效吗?
作者:互联网
我在Kotlin中为Android集成了一个简单的演示应用程序,该应用程序使用Jsoup检索了网页的标题.我正在使用Dispatchers.Main作为上下文进行网络呼叫.
我对协程的理解是,如果我在Dispatchers.Main上调用launch,它的确在主线程上运行,但是会暂停执行以免阻塞线程.
我对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.
解决方法:
My understanding of coroutines is that if I call launch on the Dispatchers.Main it does run on the main thread, but suspends the execution so as to not block the thread.
挂起执行以便不阻塞线程的唯一点是标记为暂停的方法,即挂起方法.
由于Jsoup.connect(url).get()不是挂起方法,因此它会阻塞当前线程.当您使用Dispatchers.Main时,当前线程是主线程,并且您的网络操作直接在主线程上运行,从而导致NetworkOnMainThreadException.
可以通过将其包装在withContext()中来使类似于get()方法的阻塞工作挂起,这是一个挂起方法,可确保在该方法运行时不会阻塞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
}
标签:kotlin,kotlinx-coroutines,android 来源: https://codeday.me/bug/20191108/2007570.html