[英]How to get String as a return value from "GlobalScope.launch" block
在这个应用程序中,我使用 HttpURLConnection 下载 Feed RSS“来自 XML 链接”,然后解析它并查看到 listview,但是在我运行应用程序后,我得到了空的 listview
编码
private fun downloadXML(urlPath: String? ): String {
val xmlResult = StringBuilder()
GlobalScope.launch(Dispatchers.IO) {
kotlin.runCatching {
try {
Log.d(TAG,"Dispatchers.IO Called")
val url = URL(urlPath)
val connection: HttpURLConnection = url.openConnection() as HttpURLConnection
val responseCode = connection.responseCode
Log.d(TAG, "DownloadXML: the response code is $responseCode")
connection.inputStream.buffered().reader().use { xmlResult.append(it.readText()) }
Log.e(TAG,"xmlResult: is ${xmlResult}")
}catch (ex:Exception){
when(ex){
is MalformedURLException -> ex.message.toString()
is IOException -> ex.message.toString()
is SecurityException -> ex.message.toString()
else -> ex.message.toString()
}
}
}
}
Log.e(TAG,xmlResult.toString()) //====> THIS CODEDOES NOT EXECUTED
return xmlResult.toString() // AND THIS
}
并像这样使用它
private var feedURL:String = "http://ax.itunes.apple.com/WebObjects/MZStoreServices.woa/ws/RSS/topfreeapplications/limit=10/xml"
private var feedLimit = 10
private var feedCashedURL = "INVALIDAED"
private val STATE_URL = "feedURL"
private val STATE_LIMIT = "feedLimit"
private val parseApplication = ParseApplication()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
binding = ActivityMainBinding.inflate(layoutInflater)
setContentView(binding.root)
downloadURL(feedURL.format(feedLimit))
val feedAdapter = FeedAdapter(this,R.layout.list_record,parseApplication.application)
binding.listView.adapter = feedAdapter
Log.d(TAG,"onCreate called")
}
private fun downloadURL(feedURL: String) {
if(feedURL != feedCashedURL){
Log.d(TAG, "download URL starting AsyncTask")
val rssFeed = downloadXML(feedURL)
parseApplication.parse(rssFeed)
feedCashedURL = feedURL
Log.d(TAG, "downloadURL done")
}else{
Log.d(TAG, "downloadURL URL not changed")
}
}
看起来这两行没有执行,因为即使我得到了响应代码 200,并且打印了xmlResult但仅在GlobalScope.launch
块内,我在结果中什么也没有得到,那么如何从 Coroutine 块中获取这个返回结果?
Log.e(TAG,xmlResult.toString())
return xmlResult.toString()
这两行正在执行,但不是按照您认为的顺序执行。 GlobalScope.launch()
启动异步操作,它不会等待它完成,因此return xmlResult.toString()
可能在xmlResult.append()
之前执行。
我假设您试图避免阻塞主/UI 线程。 但请注意,您的downloadURL()
仍然设计为需要等待资源下载后才能成功返回。 而且因为这个方法是从 UI 线程调用的,所以你无法避免阻塞这个线程。
为了解决这个问题,您需要将整个资源加载过程移到异步操作中。 这意味着: downloadXML()
、 downloadURL()
甚至为适配器提供数据。 只需删除您当前的GlobalScope.launch()
,然后在onCreate()
内执行以下操作:
GlobalScope.launch(Dispatchers.IO) {
val rssFeed = async { downloadXML(feedURL) }
parseApplication.parse(rssFeed.await())
withContext(Dispatchers.Main){
val feedAdapter = FeedAdapter(this@MainActivity, R.layout.list_record, parseApplication.application)
binding.listView.adapter = feedAdapter
}
}
我想这可能不是一个完全有效的示例,因为在馈送适配器时您可能需要跳回 UI 线程。 另外,我认为不推荐使用GlobalScope
,但我不太了解 Android,所以我不会提出更好的解决方案。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.