官术网_书友最值得收藏!

A synchronous function wrapped in an asynchronous caller

The first approach is quite simple. We can create a loadNews() function that directly invokes fetchRssHeadlines() and displays its results in the same way we did before:

private fun loadNews(){
val headlines = fetchRssHeadlines()
val newsCount = findViewById<TextView>(R.id.newsCount)
launch(UI) {
newsCount.text = "Found ${headlines.size} News"
}
}

Notice that this function is synchronous, and will call fetchRssHeadlines() in the same thread that it's called. Consider this code, for example:

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
loadNews()
}

Doing this will result in a NetworkOnMainThreadException being thrown. Since loadNews() uses the same thread it's being called on, the request to fetch the feed will happen in the UI thread. To fix this, we can then wrap the call to loadNews() in a launch() block like we did before, using the dispatcher that was created for the service request. The code would be almost the same:

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
launch (dispatcher) {
loadNews()
}
}

This code is quite explicit, indicating that there is some code being executed asynchronously, which is a great practice. It's also quite flexible, because if the caller of loadNews() is already in a background thread, it can fetch the news in the same background thread, without having to use a launch() or async() builder.

The disadvantage of this approach is that if we have many places calling loadNews() from the UI thread, we would have many similar blocks of launch(dispatcher) {...} spread in the code, which can make the code less readable.

主站蜘蛛池模板: 合作市| 抚远县| 曲阜市| 高尔夫| 阿勒泰市| 肥西县| 丰台区| 涟源市| 会泽县| 建平县| 昭通市| 吕梁市| 东丽区| 威海市| 祁连县| 马尔康县| 博兴县| 法库县| 赤水市| 龙山县| 准格尔旗| 东山县| 招远市| 阜平县| 海原县| 吉首市| 西吉县| 凌源市| 龙南县| 正蓝旗| 延津县| 宁强县| 东光县| 浮梁县| 九江县| 缙云县| 金门县| 上饶县| 福海县| 襄樊市| 分宜县|