不适当的阻塞方法调用,但 Suspend 函数 'withContext' 应该只从协程或另一个挂起函数中调用
Inappropriate blocking method call, but Suspend function 'withContext' should be called only from a coroutine or another suspend function
在我的服务中,我需要调用 onStartCommand
一些需要 withContext(Dispatchers.IO)
而不是 CoroutineScope(Dispatchers.IO)
的方法,例如:
- url = URL(pokemon.linkImage)
- url.openConnection().getInputStream()
- fOut= FileOutputStream(文件)
- fOut.flush()
- fOut.close()
但是Suspend function 'withContext' should be called only from a coroutine or another suspend function
。因此,如果 onStartCommand
不能成为挂起函数,因为它具有覆盖,并且 withContext
不能被 CoroutineScope 调用,因为 Inappropriate blocking method call
of methods
val url = URL(pokemon.linkImage)
val iS = url.openConnection().getInputStream()
我该如何解决这个问题?
我的onStartCommand():
override fun onStartCommand(intent: Intent, flags: Int, startId: Int): Int {
//create the directory on internal storage that will contains all pokemon images
val path = applicationContext.filesDir.absolutePath
val dirName = "/pokeimg"
val dir = File(path, dirName)
if(!dir.exists())
dir.mkdir()
CoroutineScope(Dispatchers.IO).launch {
//get the list of pokemon when they are loaded from repository
val listOfPokemon =
PersistenceSingletonRepository.getInstance(applicationContext).getListOfPokemon()
//download all the image for each pokemon in the list, but only if the image is
//not present in the directory
for(pokemon in listOfPokemon) {
val imageName = "${pokemon.name}.png"
val file = File("${dir.absolutePath}/$imageName")
//check if image not exists on internal storage and if is not an official-artwork
//images that aren't official-artwork has less quality so don't resize
if(!file.exists()) {
//download img
val url = URL(pokemon.linkImage)
val iS = url.openConnection().getInputStream()
val opts = BitmapFactory.Options()
if (!Utils.isBadImage(pokemon.linkImage)) {
//request a smaller image
opts.inSampleSize = 2
}
val bitmap = BitmapFactory.decodeStream(iS, null, opts)
val fOut= FileOutputStream(file)
bitmap?.compress(Bitmap.CompressFormat.PNG, 100, fOut)
fOut.flush()
fOut.close()
}
}
stopSelf()
}
return START_NOT_STICKY
}
您可以安全地忽略该警告,众所周知它有很多误报。您在 IO 调度程序中启动了协程,它是为阻塞调用而设计的。
另一方面,使用没有父级且没有生命周期绑定的临时 CoroutineScope
启动任何东西通常是错误的做法。您应该尊重服务生命周期并在 onDestroy
中取消协程。例如参见 [=12=].
在我的服务中,我需要调用 onStartCommand
一些需要 withContext(Dispatchers.IO)
而不是 CoroutineScope(Dispatchers.IO)
的方法,例如:
- url = URL(pokemon.linkImage)
- url.openConnection().getInputStream()
- fOut= FileOutputStream(文件)
- fOut.flush()
- fOut.close()
但是Suspend function 'withContext' should be called only from a coroutine or another suspend function
。因此,如果 onStartCommand
不能成为挂起函数,因为它具有覆盖,并且 withContext
不能被 CoroutineScope 调用,因为 Inappropriate blocking method call
of methods
val url = URL(pokemon.linkImage)
val iS = url.openConnection().getInputStream()
我该如何解决这个问题?
我的onStartCommand():
override fun onStartCommand(intent: Intent, flags: Int, startId: Int): Int {
//create the directory on internal storage that will contains all pokemon images
val path = applicationContext.filesDir.absolutePath
val dirName = "/pokeimg"
val dir = File(path, dirName)
if(!dir.exists())
dir.mkdir()
CoroutineScope(Dispatchers.IO).launch {
//get the list of pokemon when they are loaded from repository
val listOfPokemon =
PersistenceSingletonRepository.getInstance(applicationContext).getListOfPokemon()
//download all the image for each pokemon in the list, but only if the image is
//not present in the directory
for(pokemon in listOfPokemon) {
val imageName = "${pokemon.name}.png"
val file = File("${dir.absolutePath}/$imageName")
//check if image not exists on internal storage and if is not an official-artwork
//images that aren't official-artwork has less quality so don't resize
if(!file.exists()) {
//download img
val url = URL(pokemon.linkImage)
val iS = url.openConnection().getInputStream()
val opts = BitmapFactory.Options()
if (!Utils.isBadImage(pokemon.linkImage)) {
//request a smaller image
opts.inSampleSize = 2
}
val bitmap = BitmapFactory.decodeStream(iS, null, opts)
val fOut= FileOutputStream(file)
bitmap?.compress(Bitmap.CompressFormat.PNG, 100, fOut)
fOut.flush()
fOut.close()
}
}
stopSelf()
}
return START_NOT_STICKY
}
您可以安全地忽略该警告,众所周知它有很多误报。您在 IO 调度程序中启动了协程,它是为阻塞调用而设计的。
另一方面,使用没有父级且没有生命周期绑定的临时 CoroutineScope
启动任何东西通常是错误的做法。您应该尊重服务生命周期并在 onDestroy
中取消协程。例如参见 [=12=].