0

编辑:已解决。对不起,伙计们,我突然想到,虽然我的数据加载是在后台线程中发生的,但解析数据的回调却没有。数据的解析花费了很长时间,这就是锁定我的线程的原因。

编辑:我注意到我的首要问题(ProgressDialog 不再旋转)可能是由我使用 ProgressDialog 的问题引起的,而不是 UI 线程被阻塞。如果是这种情况,我该如何解决?

编辑:澄清一下,这不会永远锁定整个程序。加载完所有内容后,将关闭进度对话框,并启动新活动。我的问题是,当它正在加载时,整个 UI 被锁定(即进度对话框停止旋转)

TL;DR:doInBackground() 中的 Thread.sleep() 正在锁定 UI 线程

我有一个应用程序,当一个特定的活动打开时,它开始从我的后端在后台加载大量数据,例如,与计划相关的大量数据。该信息不会立即使用,但可以在用户尝试访问它时使用(即通过单击计划按钮并启动计划活动)。

如果用户在单击计划按钮之前稍等片刻,所有数据都会加载,计划活动将打开,并显示一切都很好。我的问题是如果他们在数据加载之前单击按钮。

我的解决方案是创建一个显示 ProgressDialog 的 ASyncTask,同时定期检查数据是否已完成加载,否则休眠。它通过一些应用程序范围的布尔变量知道数据已完成加载。我的问题是,即使 Thread.sleep() 在 doinbackground() 中运行,它仍然会锁定 UI 线程。

我正在使用自定义 ASyncTask,定义如下:

public class LoadWithProgressDialog extends AsyncTask<Void, Void, Boolean>{
    private ProgressDialog pd; //the progress dialog
    private String title; //the title of the progress dialog
    private String  message; //the body of the progress dialog
    private Runnable task; //contains the code we want to run in the background
    private Runnable postTask; //execute when the task ends
    private Context c;


    public LoadWithProgressDialog(Context context,String t, String m,Runnable r, Runnable postR){
        super();
        c = context;
        task = r;
        postTask = postR;
        title = t;
        message = m;
    }

    @Override
    protected void onPreExecute(){
        pd = ProgressDialog.show(c,title, message, false, false);
    }

    @Override
    protected Boolean doInBackground(Void... params) {
        task.run();
        return true;
    }


    @Override
    protected void onPostExecute(Boolean result) {
        pd.dismiss();
        if(postTask != null)
            postTask.run();
    }

并通过(例如)调用它:

if(loadingSchedule){

            LoadWithProgressDialog lwpd = new LoadWithProgressDialog(thisClass,"Loading","Loading Schedule", new Runnable() {
                public void run(){
                    while(loadingSchedule){
                            try {
                                Thread.sleep(1000);
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }

                    }
                }
            },new Runnable() {
                public void run(){

                    Intent i= new Intent(thisClass, Schedule.class);
                    i.putExtra("numberIneed",index);
                    startActivity(i); 
                }
            });
            lwpd.execute();

(我将全局变量的访问缩短为“loadingSchedule”以使其更易于阅读)

如何使它不锁定 UI 线程?

4

2 回答 2

5

doInBackground() 中的 Thread.sleep() 正在锁定 UI 线程

doInBackground() 中的 Thread.sleep() 不会锁定 UI 线程,正如其名称所述,任务在后台异步执行。

您的 UI 线程没有被锁定。它实际上是在 AsyncTask 启动之前弹出的 ProgressDialog(在 onPreExecute() 方法中)并一直显示在前面并在 AsyncTask 执行期间阻止用户与底层活动交互(在 doInBackground() 方法中)。最后在 AsyncTask 完成后解散。(在 onPostExecute() 方法中)

我有一个应用程序,当特定活动打开时,它开始从我的后端在后台加载大量数据

这是一个常见的用例,创建一个在后台加载数据的 AsyncTask 实现会更有效,而不是创建一个显示 ProgressDialog 的 ASyncTask,同时定期检查数据是否已完成加载,否则会休眠。

在这里查看 API ,它包含如何正确使用它的好例子。

于 2012-02-15T00:19:47.350 回答
0

您的代码看起来正确,不应阻塞主线程。

所以使用调试器!在 lwpd.execute 之后的 doInBackground、Thread.sleep、onPostExecute 中设置断点,您将了解代码的获取位置和时间。

尝试找到“UI 线程究竟在哪里被阻塞”的答案。

于 2012-02-14T23:06:55.090 回答