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

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.

主站蜘蛛池模板: 衡东县| 大英县| 乐安县| 黄平县| 彰化县| 博客| 崇义县| 阿合奇县| 咸宁市| 利津县| 茂名市| 南昌市| 织金县| 略阳县| 鸡泽县| 台南县| 饶河县| 广宁县| 南宫市| 崇礼县| 自贡市| 合阳县| 泸定县| 红河县| 兴海县| 邯郸县| 织金县| 青龙| 高碑店市| 石狮市| 于都县| 武义县| 遂昌县| 正阳县| 米易县| 临澧县| 饶平县| 韩城市| 灵寿县| 望都县| 双峰县|