在 Android 应用程序中使用 AsyncTask 在 for 循环期间更新 ProgressBar

Posted

技术标签:

【中文标题】在 Android 应用程序中使用 AsyncTask 在 for 循环期间更新 ProgressBar【英文标题】:Update ProgressBar during a for loop using AsyncTask in Android app 【发布时间】:2020-07-10 05:47:19 【问题描述】:

我正在尝试使用 AsyncTask 在 for 循环中更新 ProgressBar 的应用程序。最好的方法是向您展示我正在尝试做的伪代码

button.setOnClickListener
    for(int i=0; i<5000; i++)
    
        doSomeHeavyStuff();
        UpdateProgressBarAsyncTask(i).execute()
    

这是我目前所拥有的

主活动:

    class MainActivity : AppCompatActivity() 

    var progress:ProgressBar? = null


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

        progress = findViewById(R.id.progress)
        val buttonStart = findViewById<TextView>(R.id.button)
        var maxNumber = 5000
        buttonStart.setOnClickListener
            for(i in 0 until maxNumber)
            
                HeavyStuff()
                ProgressTask(i,progress!!,maxNumber,this).execute()
            
        


    

internal class ProgressTask (var actual:Int, var progress: ProgressBar, var max: Int, var context: Activity): AsyncTask <Void, Int, Int>()
    
        override fun onPreExecute() 
            super.onPreExecute()
            progress.visibility = View.VISIBLE
            progress.max = max
        

        override fun onProgressUpdate(vararg values: Int?) 
            super.onProgressUpdate(*values)
            progress.setProgress(values[0]!!)
        

        override fun doInBackground(vararg params: Void?): Int? 
            publishProgress(actual)
            return null
        

        override fun onPostExecute(result: Int?) 
            super.onPostExecute(result)
            progress.visibility = View.INVISIBLE
            Toast.makeText(context, "Finished!", Toast.LENGTH_SHORT).show()
        
    

XML:

<ProgressBar
        android:id="@+id/progress"
        style="@android:style/Widget.ProgressBar.Horizontal"
        android:layout_
        android:layout_
        android:layout_marginStart="24dp"
        android:layout_marginLeft="24dp"
        android:layout_marginTop="24dp"
        android:layout_marginEnd="24dp"
        android:layout_marginRight="24dp"
        android:layout_marginBottom="24dp"
        android:visibility="invisible"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

我在这里错过了什么吗?现在它在 HeavyStuff() 完成后显示 ProgressBar,因此在 for 循环期间不显示。我在这里错过了什么?

你们能帮帮我吗?

谢谢

【问题讨论】:

所有繁重的东西都需要在doBackground函数中调用。 【参考方案1】:

实际上,我认为doBackground函数中需要同时存在重物和For循环(在主线程中调用重物会冻结UI并导致ANR),请参见下面的代码:

const val max = 50000

class MainActivity : AppCompatActivity() 

    var progress: ProgressBar? = null

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

        progress = findViewById(R.id.progress)
        buttonStart.setOnClickListener 
            ProgressTask(progress!!, max, this).execute()
        
    

    internal class ProgressTask(
        var progress: ProgressBar,
        var max: Int,
        var context: Activity
    ) : AsyncTask<Void, Int, Int>() 
        override fun onPreExecute() 
            super.onPreExecute()
            progress.visibility = View.VISIBLE
            progress.max = max
        

        override fun onProgressUpdate(vararg values: Int?) 
            super.onProgressUpdate(*values)
            progress.setProgress(values[0]!!)
        

        override fun doInBackground(vararg params: Void?): Int? 
            for (i in 0 until max) 
                doHeavyStuff()
                publishProgress(i)
            
            return null
        

        override fun onPostExecute(result: Int?) 
            super.onPostExecute(result)
            progress.visibility = View.INVISIBLE
            Toast.makeText(context, "Finished!", Toast.LENGTH_SHORT).show()
        
    

【讨论】:

非常感谢。这很有帮助:) 很高兴听到这个消息【参考方案2】:

我会采用这种方法来避免内存泄漏:

private lateinit var mTextView: WeakReference<TextView>
private lateinit var mProgressBar: WeakReference<ProgressBar>

private const val MAX = 50000

class ProgressTask(
    pb: ProgressBar,

var max: Int
) : AsyncTask<Void, Int, Int>() 

init 
    mProgressBar = WeakReference(pb)


override fun onPreExecute() 
    super.onPreExecute()
    mProgressBar.get()?.visibility = View.VISIBLE
    mProgressBar.get()?.max = MAX


override fun onProgressUpdate(vararg values: Int?) 
    super.onProgressUpdate(*values)
    mProgressBar.get()?.progress = values[0]!!


override fun doInBackground(vararg p0: Void?): Int 
    for (i in 0 until MAX) 
        doHeavyStuff()
        publishProgress(i)
    
    return 1


override fun onPostExecute(result: Int?) 
    super.onPostExecute(result)
    mProgressBar.get()?.visibility = View.GONE



【讨论】:

以上是关于在 Android 应用程序中使用 AsyncTask 在 for 循环期间更新 ProgressBar的主要内容,如果未能解决你的问题,请参考以下文章

AsyncTask介绍

当片段视图加载是异步任务的一部分时,如何在片段加载之前显示进度条?

如何在 android 应用程序中使用 OSM 地图。?有啥教程可以学习在android中使用OSM吗?

我可以在 sencha 应用程序(Android)中使用 android AsyncTask

在 Android Studio 中使用 ActivityInstrumentationTestCase2 在横向测试 Android 应用程序

如何在 Android 应用程序中使用自定义主题