我正试图将iPhone应用程序移植到Android,但在启动时遇到了一些困难。
我有以下功能来获取数据:
public void onClick(View v) {
try{
//Toast.makeText(getApplicationContext(), "Starting update", Toast.LENGTH_SHORT).show();
progressDialog = ProgressDialog.show(MainActivity.this, "", "Loading...");
new Thread() {
public void run() {
try{
DataManager manager = new DataManager(getApplicationContext());
manager.updateData();
} catch (Exception e) {
Log.e(TAG, e.getMessage());
AlertDialog alertDialog = new AlertDialog.Builder(MainActivity.this).create();
alertDialog.setTitle("Error...");
alertDialog.setMessage(e.getMessage());
alertDialog.show();
}
progressDialog.dismiss();
}
}.start();
} catch (Exception e) {
Log.e(TAG, "Problem with getting the data: " + e.getMessage());
Toast.makeText(getApplicationContext(), "Error during update",
Toast.LENGTH_SHORT).show();
}
}
但如果发生异常,我会得到以下错误(我最初试图使用AsyncTask,但遇到了同样的问题):
08-29 20:23:24.389: ERROR/WindowManager(974): Activity org.idoms.iDomsAndroid.MainActivity has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@412fcc38 that was originally added here
android.view.WindowLeaked: Activity org.idoms.iDomsAndroid.MainActivity has leaked window com.android.internal.policy.impl.PhoneWindow$DecorView@412fcc38 that was originally added here
at android.view.ViewRootImpl.<init>(ViewRootImpl.java:344)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:267)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:215)
at android.view.WindowManagerImpl$CompatModeWrapper.addView(WindowManagerImpl.java:140)
at android.view.Window$LocalWindowManager.addView(Window.java:537)
at android.app.Dialog.show(Dialog.java:278)
at android.app.ProgressDialog.show(ProgressDialog.java:116)
at android.app.ProgressDialog.show(ProgressDialog.java:99)
at android.app.ProgressDialog.show(ProgressDialog.java:94)
at org.idoms.iDomsAndroid.MainActivity$1.onClick(MainActivity.java:46)
at android.view.View.performClick(View.java:3511)
at android.view.View$PerformClick.run(View.java:14105)
at android.os.Handler.handleCallback(Handler.java:605)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:4424)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
at dalvik.system.NativeStart.main(Native Method)
做这件事的正确方法是什么?
我的猜测是,线程不能与主UI线程元素交互。
progressDialog是在主Ui线程中创建的,其他线程无法与它交互。你可以在字段中给它一个信号并检查它的进度,但我真的建议使用Asynctask来完成这项任务。
异步任务
UI元素(如AlertDialog)只能从主(UI)线程进行操作。如果您试图从后台线程操作它们(这就是您使用new Thread() { ... }
所做的),您将得到此Exception。
您可以通过在访问AlertDialog
或progressDialog
时调用runOnUiThread(new Runnable() { ... });
来解决此问题,但这是非常非常脏的。
更好的解决方案是使用AsyncTask
(或者更好的AsyncTaskLoader
,因为它可以为您处理方向变化和其他一些事情)。使用AsyncTask时,请确保不访问doInBackground()
-方法中的UI元素。作为名称,saysdoInBackground(...)
在后台线程中执行。如果您试图通过该方法操作UI元素,您确实会遇到与现在相同的错误。如果要从AsyncTask
访问UI元素,可以使用onPreExecute(...)
、onProgressUpdate(...)
和onPostExecute(...)
方法进行访问。
窗口泄漏的原因是您的alertDialog在应用程序关闭前没有被关闭。