Обновите ProgressBar в течение для l oop, используя AsyncTask в приложении Android - PullRequest
0 голосов
/ 30 марта 2020

Я пытаюсь создать приложение, в котором я хочу обновить ProgressBar в a для l oop, используя AsyncTask. Лучше всего показать вам псевдокод того, что я пытаюсь сделать

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

Это то, что я имею до сих пор

MainActivity:

    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_width="300dp"
        android:layout_height="wrap_content"
        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" />

Я что-то здесь пропускаю? Прямо сейчас он показывает ProgressBar после того, как HeavyStuff () закончен, и, таким образом, он не отображается в течение l oop. Что мне здесь не хватает?

Ребята, не могли бы вы помочь мне с этим?

Спасибо

1 Ответ

1 голос
/ 30 марта 2020

На самом деле, я думаю, что и тяжелые вещи, и For L oop должны присутствовать внутри функции doBackground (вызов тяжелых вещей в главном потоке замораживает пользовательский интерфейс и вызывает 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()
        }
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...