进度条与异步任务一起使用

10

我想在屏幕上使用进度条而不是进度对话框。

我已经在XML视图文件中插入了一个进度条,我希望在加载时显示它,并在不加载时禁用它。

因此,我使用了“visible”,但它会占用空间,导致其他数据下移。

我应该如何在AsyncTask中使用进度条?如何显示和隐藏它?


可能是从AsyncTask更新Activity中的进度对话框的重复问题。 - Siddharth Lele
3个回答

26

以下是一个最详尽的示例:

public class ScreenSplash extends Activity {

    @Override
    protected void onCreate(final Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.screen_splash);

        final ProgressBar progress = (ProgressBar)findViewById(R.id.progress);
        final TextView    textview = (TextView)findViewById(R.id.text);

        new MyWorker(this, progress, textview).execute();
    }
}


final class MyWorker extends AsyncTask<Void, Integer, Void> {
    private static final int titles[] = {R.string.splash_load_timezone,
                                         R.string.splash_load_memory,
                                         R.string.splash_load_genres,
                                         R.string.splash_load_channels,
                                         R.string.splash_load_content};
    private static final int progr[]  = {30, 15, 20, 25, 20};

    private int index;

    private final Activity parent;
    private final ProgressBar progress;
    private final TextView textview;

    public MyWorker(final Activity parent, final ProgressBar progress, final TextView textview) {
        this.parent = parent;
        this.progress = progress;
        this.textview = textview;
    }

    @Override
    protected void onPreExecute() {
        int max = 0;
        for (final int p : progr) {
            max += p;
        }
        progress.setMax(max);
        index = 0;
    }

    @SuppressWarnings("unchecked")
    @Override
    protected Void doInBackground(final Void... params) {
        /* Load timezone. this is very slow - may take up to 3 seconds. */
          ...
        publishProgress();

        /* Get more free memory. */
          ...
        publishProgress();

        /* Load channels map. */
          ...
        publishProgress();

        /* Load genre names. */
          ...
        publishProgress();


        /* Preload the 1st screen's content. */
          ...
        publishProgress();

        return null;
    }

    @Override
    protected void onProgressUpdate(final Integer... values) {
        textview.setText(titles[index]);
        progress.incrementProgressBy(progr[index]);
        ++index;
    }

    @Override
    protected void onPostExecute(final Void result) {
        parent.finish();
    }
}

要显示/隐藏进度条,使用progress.setVisibility(View.VISIBLE)progress.setVisibility(View.GONE)。还有View.INVISIBLE常量;它与GONE的区别在于进度条不会被绘制,但仍占据其空间(对于某些布局很有用)。


3
就像这个领域中的许多示例一样,它没有考虑到在任务运行期间活动可能被销毁(然后可能重新创建)的可能性... - Reuben Scratton
是的,没错。关于这个问题有一篇很好的帖子:https://dev59.com/YXA75IYBdhLWcg3wUHWQ - JBM
2
事实上,在onDestroy中必须杀死后台任务。Activity指南表明,在Activity被销毁后,不应该有任何后台运行的任务(否则会耗尽手机电池)。因此,即使只是用户翻转了手机并在onCreate中重新开始,您也应该强制完成AsyncTasks - 是的,这很愚蠢,但如果Google以这种方式放置了Activity生命周期,我们能做什么呢? - JBM
私有静态最终整数数组progr[] = {30, 15, 20, 25, 20}表示什么?请解释一下。。如何解释? - Zar E Ahmer
这种解决方案在旋转的情况下可能行不通,你觉得呢? - fcracker79
1
@fcracker79,是的,这个例子不支持旋转。实际上,一般来说使用AsyncTask是一个不好的想法。我的经验表明,将逻辑与UI解耦会得到最佳结果。在这种方法中,逻辑可以无需考虑Activity生命周期完成任务,而UI则可以在准备好后显示结果。 - JBM

4

当使用异步任务中的进度条时,您可以使用doInBackground(int Progress)中的PostPrgressUpdate()方法,并在OnProgressUpdate()方法中相应地更新进度条。至于进度条的显示和隐藏,我无法理解您的问题(或问题,请修正)。


-1

在 XML 中使用

<ProgressBar
        android:id="@+id/progress_bar"
        style="@android:style/Widget.ProgressBar.Small"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        />

然后在Asyntask中

 ProgressBar mProgress=(ProgressBar) findViewById(R.id.progress_bar);

 @Override
    protected void onPostExecute(String s) {
        super.onPostExecute(s);
        mProgress.setVisibility(View.GONE);
    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接