Я начинаю думать, что, чтобы заставить ProgressDialog работать AsyncTask должен быть внутренним классом в классе Действия. Верный? [Отредактированный намного позже... ответ является Ложью, и я не уверен, является ли это ошибкой или что. Я использую Android 1.5. Я собираюсь читать на Сервисах.]
У меня есть действие использование база данных для управления информацией. Если база данных заполняется, все хорошо. Если это не заполняется затем, я должен загрузить информацию с веб-сайта, заполнить базу данных, то доступ заполненная база данных для завершения Представлений в onCreate.
Проблема без некоторых средств определить, когда поток AsyncTask закончил заполнять базу данных, я получаю следующее сообщение Ошибки при закрытии Силы:Прошу прощения! приложение неожиданно остановилось. Я нажимаю на кнопку Force Close, фон, поток AsyncTask продолжает работать, база данных заполняется, и все работает хорошо.
Я должен избавиться от того сообщения об ошибке и нуждаться в некоторой помощи о том, как сделать это. Вот некоторый код psuedo:
public class ViewStuff extends Activity
{
onCreate
{
if(database is populated)
do_stuff
else
{
FillDB task = null;
if(task == null || task.getStatus().equals(AsyncTask.Status.FINISHED))
{
task = new FillDB(context);
task.execute(null);
}
}
continue with onCreate using information from database to properly display
} // end onCreate
} // end class
В отдельном файле:
public class FillDB extends AsyncTask<Void, Void, Void>
{
private Context context;
public FillDB (Context c) //pass the context in the constructor
{
context = c;
}
public void filldb ()
{
doInBackground();
}
@Override
protected void onPreExecute()
{
ProgressDialog progressDialog = new ProgressDialog(context);
//crashes with the following line
progressDialog.show(context, "Working..", "Retrieving info");
}
@Override
protected Void doInBackground(Void... params)
{
// TODO Auto-generated method stub
try
etc etc etc
}
}
Вот отслеживание стека от эмулятора:
----- pid 846 at 2010-03-21 19:58:25 -----
Cmd line: com.trial
DALVIK THREADS:
"main" prio=5 tid=3 NATIVE
| group="main" sCount=1 dsCount=0 s=0 obj=0x40018e70
| sysTid=846 nice=0 sched=0/0 handle=-1098855268
at android.os.BinderProxy.transact(Native Method)
at android.app.ActivityManagerProxy.handleApplicationError(ActivityManagerNative.java:2103)
at com.android.internal.os.RuntimeInit.crash(RuntimeInit.java:302)
at com.android.internal.os.RuntimeInit$UncaughtHandler.uncaughtException(RuntimeInit.java:75)
at java.lang.ThreadGroup.uncaughtException(ThreadGroup.java:887)
at java.lang.ThreadGroup.uncaughtException(ThreadGroup.java:884)
at dalvik.system.NativeStart.main(Native Method)
"Binder Thread #3" prio=5 tid=15 NATIVE
| group="main" sCount=1 dsCount=0 s=0 obj=0x43733d88
| sysTid=852 nice=0 sched=0/0 handle=1486928
at dalvik.system.NativeStart.run(Native Method)
"Binder Thread #2" prio=5 tid=13 NATIVE
| group="main" sCount=1 dsCount=0 s=0 obj=0x437313c8
| sysTid=851 nice=0 sched=0/0 handle=1492472
at dalvik.system.NativeStart.run(Native Method)
"Binder Thread #1" prio=5 tid=11 NATIVE
| group="main" sCount=1 dsCount=0 s=0 obj=0x4372b9b0
| sysTid=850 nice=0 sched=0/0 handle=1492664
at dalvik.system.NativeStart.run(Native Method)
"JDWP" daemon prio=5 tid=9 VMWAIT
| group="system" sCount=1 dsCount=0 s=0 obj=0x4372a2a0
| sysTid=849 nice=0 sched=0/0 handle=1490176
at dalvik.system.NativeStart.run(Native Method)
"Signal Catcher" daemon prio=5 tid=7 RUNNABLE
| group="system" sCount=0 dsCount=0 s=0 obj=0x4372a1e8
| sysTid=848 nice=0 sched=0/0 handle=1487888
at dalvik.system.NativeStart.run(Native Method)
"HeapWorker" daemon prio=5 tid=5 VMWAIT
| group="system" sCount=1 dsCount=0 s=0 obj=0x427d03c0
| sysTid=847 nice=0 sched=0/0 handle=1487592
at dalvik.system.NativeStart.run(Native Method)
----- end 846 -----
Что я делаю неправильно?
Г-н Snowflake,
Попробованный:
@Override
protected void onPreExecute()
{
Activity.this.runOnUiThread(new Runnable() {
public void run() {
ProgressDialog progressDialog = new ProgressDialog(context);
//crashes with the following line
progressDialog.show(context, "Working..", "Retrieving info");
}
});
}
Activity.this отмечается как ошибка: Никакой экземпляр включения Действия типа не доступен в объеме
Я думаю, что мне нужно к FillDB, расширяется, Действие затем создают частный класс в рамках расширения FillDB AsyncTask? Та привычка' работа. Никакое обеспечение, когда действие FillDB запустится и не может использовать startActivityForResult, так как никакой результат не возвращается из AsyncTask, когда это завершается.
Обновление: Испытанное создание частного класса в классе вызова. Все еще не может показать ProgressDialog. Одна из ошибок: Не мог добавить окно - маркерный пустой указатель не для приложения. Я понятия не имею, какой маркер упоминается.
Я отказался от AsyncTask и использовал обработчик для обработки потока. Все хорошо.
Не уверен, что это ошибка в версии 1.5 с AsyncTask или что-то еще, чего я не делал.
Рассматривали ли вы возможность переопределения метода onCreateDialog () для реализации ваших диалогов? Таким образом их легко создавать и контролировать, поскольку Android делает большую часть работы за вас. У меня есть пример ниже, делающий именно то, что вы пытаетесь сделать.
Попробуйте запустить
ProgressDialog progressDialog = new ProgressDialog(context);
//crashes with the following line
progressDialog.show(context, "Working..", "Retrieving info");
как запускаемый аргумент Activity.runOnUiThread (). Элементы пользовательского интерфейса должны создаваться и управляться в потоке пользовательского интерфейса. Думаю, это также верно для Dialog
s.
[edit] Код:
Activity.this.runOnUiThread(new Runnable() {
public void run() {
ProgressDialog progressDialog = new ProgressDialog(context);
//crashes with the following line
progressDialog.show(context, "Working..", "Retrieving info");
}
};
У меня была та же проблема, что и у OP, в течение некоторого времени с публичной (внешней) AsyncTask. Исследуя сообщение об ошибке "attempted to add window with non-application token", я обнаружил, что проблема заключается в установке контекста, передаваемого в конструкторе AsyncTask. Контекст, который должен быть передан, это "this", а не this.getApplicationContext() или this.getBaseContext(). Это решило мою проблему.