2

所以我有这些长时间运行的服务器调用,这些调用基本上是在我的 Android 应用程序中进行的 OData。调用的使用者.execute().get()用来等待来自网络线程的响应(我知道正确的方法是使整个事情异步和基于回调,但是如果没有这些数据,应用程序将无法以任何方式运行,并且完全重新架构它以以这种方式工作似乎没有任何好处)。

因此,我在网上找到了许多片段,其中ProgressDialog结合使用onPreExecute()onPostExecute(),如下面的代码示例所示,可用于在AsyncTask执行时显示进度对话框。我正在使用提供的示例,但发生的情况是调用开始,它等待网络事务,然后很快闪烁并隐藏进度对话框。它可以在事务上等待整整几秒钟,我知道它正在等待doInBackground(),但对话框直到最后才会弹出,使其实际上毫无用处。

在下面的代码中,DoEvents()bit 基本上只是一个非常短暂的睡眠。我试过有没有它,似乎没有区别,但似乎值得一试。

class GetFromServerTask extends AsyncTask<String, Void, String>
    {
        private Context context;
        private ProgressDialog dialog;

        public GetFromServerTask(Context ctx) {
            context = ctx;
            dialog = new ProgressDialog(ctx);
        }

        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            dialog.setMessage("Loading...");
            dialog.show();
            DoEvents();
        }

        @Override
        protected String doInBackground(String... parms) {
            if(InOfflineMode)
                return "notdeserializable";

            String url = parms[0]; 
            HttpURLConnection urlConnection = null;
            try {
                URL typedUrl = new URL(url);
                urlConnection = (HttpURLConnection) typedUrl.openConnection();

                //Add Authorization token
                if(InDebugMode) {
                    urlConnection.addRequestProperty("AuthToken", AuthToken);
                } else {
                    urlConnection.addRequestProperty("Authorization", "Bearer " + AuthToken);
                }
                urlConnection.addRequestProperty("Accept", "application/json");

                DoEvents();
                InputStream in = new BufferedInputStream(urlConnection.getInputStream());
                byte[] contents = new byte[in.available()];

                int bytesRead = 0;
                String strContents = ""; 
                while((bytesRead = in.read(contents)) != -1){ 
                    strContents += new String(contents, 0, bytesRead);
                    DoEvents();
                }

                if(strContents.startsWith("<HTML>"))
                    return "Error: Received unexpected HTML when connecting to service.  Make sure you are not connected to a WIFI that requires authentication.";

                return strContents;
            } catch(UnknownHostException hex) {
                return "Error: Could not find server address.  Make sure you are connected to the internet.  If you just changed connections (ie: turning WIFI on or off) it make take a minute to refresh";
            }
            catch(Exception ex) {
                String msg = "Error: " + ex.getClass().getName() + ": " + ex.getMessage();
                Log.e("TE", msg);
                return msg;
            } finally {
                if(urlConnection != null)
                    urlConnection.disconnect();
            }
        }

        @Override
        protected void onPostExecute(String result) {
            super.onPostExecute(result);
            if(dialog != null && dialog.isShowing())
                dialog.dismiss();
            DoEvents();
        }
    }

我还尝试了 SO 上其他地方建议的略有不同的版本(如下所示),结果完全相同:

protected void onPreExecute() {
    dialog=ProgressDialog.show(context, "", "Loading...", true, false);
    super.onPreExecute();
}

我还尝试将ProgressDialog所有AsyncTask内容放在一起并将其显示在任务“外部”,如下所示。在这种情况下,它甚至不会出现。

ProgressDialog dialog = ProgressDialog.show(ServerAccessLayer.m_context, "", "Loading...", true, false);
String retVal = new GetFromServerTask(ServerAccessLayer.m_context).execute(url).get();
dialog.dismiss();

return retVal; 
4

2 回答 2

3

好的,你的问题是你的.get(). .get是阻塞调用。这意味着您不会返回到事件循环(Android 框架中调用onCreate, onPause, 事件处理程序, onPostExecute,onPreExecute等的代码),直到它返回。如果您不返回事件循环,您将永远不会进入绘图代码,也不会显示进度对话框。如果要显示对话框,则需要重新架构应用程序以实际异步使用任务。旁注——如果你.get()在你的 UI 线程上这样调用,你的整个应用程序将冻结并且看起来很糟糕。这就是为什么他们一开始就强迫人们不要在 UI 线程上进行网络 IO。

于 2013-06-03T21:41:03.243 回答
1

正如@Gabe-Sechan 所说,这.get()就是您的 UI 冻结并且 ProgressDialog 未按您想要的那样显示的原因。另一方面,当他说:

如果要显示对话框,则需要重新架构应用程序以实际异步使用任务。

如果您只是删除您的对话框并在 中显示并关闭.get()对话框,您将得到您正在寻找的内容:任务执行时显示的 ProgressDialog。onPreExecute()onPostExecute(String result)

更新:

我基于使用Android 的 AsyncTask 类的方法的意义/有用性开始了一个问题。get()

于 2013-06-03T22:31:43.360 回答