【问题标题】:"Busy" progress dialog causes strange code processing“忙碌”进度对话框导致奇怪的代码处理
【发布时间】:2011-06-12 19:46:42
【问题描述】:

让我为你总结一下情况:

  • 我有一个按钮 (btnChooseEp),当您按下它时会出现一个 AlertDialog。
  • 在 AlertDialog 中选择某些内容时,必须评估三个 if 语句。
  • 在评估它们时,会出现一个 ProgressDialog。它表示应用程序“忙碌”。
  • 评估这些语句后,ProgressDialog 必须消失。

代码下方描述了我的问题。

这里显示了整个代码块:

ProgressDialog getTracklistProgressDialog = null;

...

    Button btnChooseEp = (Button)findViewById(R.id.btnChooseEp);
    btnChooseEp.setOnClickListener(new OnClickListener()
    {           
        public void onClick(View v)
        {               
            final AlertDialog.Builder builder = new AlertDialog.Builder(GA.this);
            builder.setTitle(getText(R.string.chooseEpisode));
            builder.setItems(episodes, new DialogInterface.OnClickListener()
            {
                public void onClick(DialogInterface dialog, final int pos)
                {                       
                    getTracklistProgressDialog = ProgressDialog.show(GA.this, "Please wait...",
                            "Retrieving tracklist...", true);

                    new Thread()
                    {
                        public void run()
                        {
                            try
                            {                       
                                String str1, epURL;

                                if(pos < 9)
                                {
                                    str1 = getResources().getString(R.string.epNo1);        
                                    epURL = String.format(str1, pos+1);
                                    setTracklist(epURL, tracklist);                 
                                }
                                else if(pos < 100)
                                {
                                    str1 = getResources().getString(R.string.epNo2);
                                    epURL = String.format(str1, pos+1);
                                    setTracklist(epURL, tracklist);
                                }
                                else if(pos >= 100)
                                {
                                    str1 = getResources().getString(R.string.epNo3);
                                    epURL = String.format(str1, pos+1);
                                    setTracklist(epURL, tracklist);
                                }
                            }
                            catch(Exception e)
                            {}

                            // Remove progress dialog
                            getTracklistProgressDialog.dismiss();                               
                        }
                    }.start();                          
                }
            });
            AlertDialog alertDialog = builder.create();
            alertDialog.show();
        }
    });

不确定是否需要,但这里是函数 setTracklist 的代码:

public void setTracklist(String string, TextView tv)
{
    try
    {
        tv.setText(getStringFromUrl(string));
    }
    catch(Exception e)
    {
        e.printStackTrace();
    } 
}

函数getStringFromUrl的代码可以看这里:http://pastebin.com/xYt3FwaS

现在,问题来了: 回到当我没有实现 ProgressDialog 的事情(我从这里得到的,顺便说一句:http://www.anddev.org/tinytut_-_displaying_a_simple_progressdialog-t250.html)时,它工作得很好 - setTracklist 函数从 Internet 上的文本文件中检索字符串,并将文本设置为 TextView。现在,当我添加 ProgressDialog 代码并将原始代码放入 try 语句时,只有极少部分的文本文件被添加到 TextView - 大约 22-24 个字符,不多。 “忙碌”的 ProgressDialog 显示得很好。它以前工作得很好;它能够将超过 1300 个字符加载到 TextView 中。 我不知道它是否与线程有关 - 我已经谷歌了很多,但没有找到答案。

那么,我如何让它加载所有数据而不是只加载一小部分?

(顺便说一句,我希望能够在所有 if 语句下方设置“setTracklist(epURL,tracklist);”行,但随后它说它无法解析“epURL”。似乎很愚蠢同一行写 3 次!)

25/1 更新了当前代码:

final Handler uiHandler = new Handler();
final Button btnChooseEp = (Button)findViewById(R.id.btnChooseEp);
btnChooseEp.setEnabled(false);
btnChooseEp.setOnClickListener(new OnClickListener()
{           
        public void onClick(View v)
        {               
            builder.setTitle(getText(R.string.chooseEpisode));
            builder.setItems(episodes2, new DialogInterface.OnClickListener()
            {
                public void onClick(DialogInterface dialog, final int pos)
                {                       
                    replay.setVisibility(View.VISIBLE);
                    replayWeb.setVisibility(View.VISIBLE);

                    getTracklistProgressDialog = ProgressDialog.show(GA.this, "Please wait...",
                            "Retrieving tracklist...", true);

                    new Thread()
                    {
                        public void run()
                        {
                            try
                            {                       
                                String str1, epURL;

                                if(pos < 9)
                                {
                                    str1 = getResources().getString(R.string.gaEpNo1);      
                                    epURL = String.format(str1, pos+1);
                                    setTracklist2(epURL, tracklist);                    
                                }
                                else if(pos < 100)
                                {
                                    str1 = getResources().getString(R.string.gaEpNo2);
                                    epURL = String.format(str1, pos+1);
                                    setTracklist2(epURL, tracklist);
                                }
                                else if(pos >= 100)
                                {
                                    str1 = getResources().getString(R.string.gaEpNo3);
                                    epURL = String.format(str1, pos+1);
                                    setTracklist2(epURL, tracklist);
                                }
                            }
                            catch(Exception e)
                            {}

                            // Remove progress dialog
                            uiHandler.post(new Runnable()
                            { 
                                public void run()
                                { 
                                    getTracklistProgressDialog.dismiss();                                       
                                }
                            }
                            );                              
                        }
                    }.start();                          
                }
            });
            AlertDialog alertDialog = builder.create();
            alertDialog.show();
        }
    });

public void setTracklist2(final String string, final TextView tv)
{
    try
    {
        uiHandler.post(
                new Runnable()
                {
                    public void run()
                    {
                        try
                        {
                            tv.setText(getStringFromUrl(string));
                        }
                        catch (UnsupportedEncodingException e)
                        {
                            e.printStackTrace();
                        }
                    }
                });
    }
    catch(Exception e)
    {
        e.printStackTrace();
    } 
}

注意:“replay”和“replayWeb”只是两个TextView。 “btnChooseEp”在按下另一个按钮时启用。

【问题讨论】:

    标签: android android-alertdialog progressdialog


    【解决方案1】:

    我的猜测是你得到了奇怪的行为,因为你在非 ui 线程上调用 ui 方法。 getTracklistProgressDialog.dismiss();
    必须在 ui 线程上执行。我的猜测是它正在崩溃,您的线程正在崩溃,然后您的一些资源处于不良状态。这可以解释为什么你会得到不同数量的字符。

    我会尝试在您的 onCreate 方法中创建一个最终处理程序,该处理程序将绑定到 uiThread。在该线程中,您可以调用

    uiHandler.post( 
     new Runnable() { 
      public void run(){ 
        getTracklistPRogressDialog.dismiss();
      }
     }
    );
    

    这很快,所以它可能在语法上不正确,请检查您的 ide。

    这是我能从你发布的内容中得到的最好的。如果您发布更多代码,我可以尝试运行它来为您提供更多帮助。

    更新:

    我想我发现了你的问题:

    拥有另一个线程的想法是在那里进行长时间运行的工作,但我们现在拥有的实际上是在 ui 线程上进行长时间运行的工作,这与我们的目标相反。我们需要做的是将调用从 setTracklist 调用移至线程中的getStringFromUrl(url)。我会重写 setTracklist 如下:

    public void setTracklist(String tracklistContent, TextView tv)
    {
        try
        {
          runOnUiThread(
            new Runnable() {
              public void run() {
                tv.setText(tracklistContent);
              }
             });
        }
        catch(Exception e)
        {
            e.printStackTrace();
        } 
    }
    

    然后在你的内部 onClick 方法中,这样做:

                public void onClick(DialogInterface dialog, final int pos)
                {                       
                    getTracklistProgressDialog = ProgressDialog.show(GA.this, "Please wait...",
                            "Retrieving tracklist...", true);
    
                    new Thread()
                    {
                        public void run()
                        {
                            try
                            {                       
                                String str1, epURL;
    
                                if(pos < 9)
                                {
                                    str1 = getResources().getString(R.string.epNo1);        
                                    epURL = String.format(str1, pos+1);
                                    String tlContent = getStringFromUrl(epUrl);
                                    setTracklist(epURL, tracklist);                 
                                }
                                else if(pos < 100)
                                {
                                    str1 = getResources().getString(R.string.epNo2);
                                    epURL = String.format(str1, pos+1);
                                    String tlContent = getStringFromUrl(epUrl);
                                    setTracklist(epURL, tracklist);
                                }
                                else if(pos >= 100)
                                {
                                    str1 = getResources().getString(R.string.epNo3);
                                    epURL = String.format(str1, pos+1);
                                    String tlContent = getStringFromUrl(epUrl);
                                    setTracklist(epURL, tracklist);
                                }
                            }
                            catch(Exception e)
                            {}
    
                            // Remove progress dialog
                            getTracklistProgressDialog.dismiss();                               
                        }
                    }.start();                          
                }
    

    我是这样,我们在重新获得 ui 线程执行之前调用 web 服务/url。长时间运行的互联网检索在 bg 线程上运行,然后 ui 更新发生在 ui 线程上。认为这应该会有所帮助。

    【讨论】:

    • 以前从未处理过Handles,但这是我所做的:Put "final Handler uiHandler = new Handler();"在 onCreate 方法中(就在 setContentView 之后)。将您的代码而不是单个 .dismiss() 语句放在我的代码中。 IDE中没有错误,但恐怕在应用程序中没有任何区别。您认为还有哪些代码可以提供帮助?
    • 您可以在工作台上的模拟器上以调试模式运行它吗?它应该在出现异常时自动中断,这将告诉我们导致问题的行。我还看到您正在 setTracklist 中设置 ui 元素的文本值,因此您应该将其包装在处理程序中,就像您进行解雇一样。所有 ui 元素的触摸都需要发生在同一个线程上
    • 事实上,我会先包装你的 setTracklist 元素,因为你吞下了所有的异常,我猜这会导致一个调用异常,它会被你的代码吃掉。
    • 如果我把“setTracklist(epURL, tracklist);”在 uiHandler 的代码中,它说无法解析 epURL。 (实际上在我第一篇文章的最后两行中写的内容相同。)我不太明白您在上一个答案中“包装 setTracklist 元素”的意思。
    • 好的,现在好像可以了。我做了什么:在代码顶部定义了字符串变量“epURL”:“static String epURL;” (在 onCreate 方法之上)。删除“setTracklist(epURL, tracklist);”从每个 if 语句中提取并将其放在 uiHandler 线程中。不确定这是否是最好的方法?
    猜你喜欢
    • 2013-01-30
    • 2013-10-26
    • 2013-05-23
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-01-07
    • 1970-01-01
    相关资源
    最近更新 更多