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

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.

主站蜘蛛池模板: 兴和县| 尼勒克县| 宣威市| 宜兰县| 横山县| 兴安盟| 兴业县| 扶沟县| 磴口县| 陆丰市| 茌平县| 台州市| 米易县| 苏州市| 光山县| 望都县| 巴塘县| 徐汇区| 绿春县| 德清县| 夏邑县| 会宁县| 政和县| 玛曲县| 彭水| 界首市| 邵阳县| 龙胜| 天祝| 化德县| 香河县| 开江县| 新疆| 平武县| 永胜县| 波密县| 阿合奇县| 河东区| 伊宁县| 合山市| 罗源县|