Android: ProgressDialog.show() аварийно завершает работу с getApplicationContext
Я не могу понять, почему это происходит. Этот код:
mProgressDialog = ProgressDialog.show(this, "", getString(R.string.loading), true);
работает просто отлично. Тем не менее, этот код:
mProgressDialog = ProgressDialog.show(getApplicationContext(), "", getString(R.string.loading), true);
выдает следующее исключение:
W/WindowManager( 569): Attempted to add window with non-application token WindowToken{438bee58 token=null}. Aborting.
D/AndroidRuntime( 2049): Shutting down VM
W/dalvikvm( 2049): threadid=3: thread exiting with uncaught exception (group=0x4001aa28)
E/AndroidRuntime( 2049): Uncaught handler: thread main exiting due to uncaught exception
E/AndroidRuntime( 2049): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.tastekid.TasteKid/com.tastekid.TasteKid.YouTube}: android.view.WindowManager$BadTokenException: Unable to add window -- token null is not for an application
E/AndroidRuntime( 2049): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2401)
E/AndroidRuntime( 2049): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2417)
E/AndroidRuntime( 2049): at android.app.ActivityThread.access$2100(ActivityThread.java:116)
E/AndroidRuntime( 2049): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1794)
E/AndroidRuntime( 2049): at android.os.Handler.dispatchMessage(Handler.java:99)
E/AndroidRuntime( 2049): at android.os.Looper.loop(Looper.java:123)
E/AndroidRuntime( 2049): at android.app.ActivityThread.main(ActivityThread.java:4203)
E/AndroidRuntime( 2049): at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime( 2049): at java.lang.reflect.Method.invoke(Method.java:521)
E/AndroidRuntime( 2049): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:791)
E/AndroidRuntime( 2049): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:549)
E/AndroidRuntime( 2049): at dalvik.system.NativeStart.main(Native Method)
E/AndroidRuntime( 2049): Caused by: android.view.WindowManager$BadTokenException: Unable to add window -- token null is not for an application
E/AndroidRuntime( 2049): at android.view.ViewRoot.setView(ViewRoot.java:460)
E/AndroidRuntime( 2049): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:177)
E/AndroidRuntime( 2049): at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:91)
E/AndroidRuntime( 2049): at android.app.Dialog.show(Dialog.java:238)
E/AndroidRuntime( 2049): at android.app.ProgressDialog.show(ProgressDialog.java:107)
E/AndroidRuntime( 2049): at android.app.ProgressDialog.show(ProgressDialog.java:90)
E/AndroidRuntime( 2049): at com.tastekid.TasteKid.YouTube.onCreate(YouTube.java:45)
E/AndroidRuntime( 2049): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1123)
E/AndroidRuntime( 2049): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2364)
E/AndroidRuntime( 2049): ... 11 more
Есть идеи, почему это происходит? Я звоню это из onCreate
метод.
18 ответов
Какую версию API вы используете? Если я прав в том, в чем проблема, то это было исправлено в Android 1.6 (версия API 4).
Похоже, что ссылка на объект getApplicationContext()
возвращает просто указывает на ноль. Я думаю, что у вас есть проблема, похожая на ту, что была у меня в том, что некоторый код в onCreate()
выполняется до того, как окно действительно будет построено. Это будет взломом, но попробуйте запустить новый поток за несколько сотен миллисекунд (IIRC: 300-400, кажется, мне подходит, но вам нужно будет повозиться), который открывает ваш ProgressDialog и запускает все остальное, что вам нужно (например, сеть IO). Что-то вроде этого:
@Override
public void onCreate(Bundle savedInstanceState) {
// do all your other stuff here
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
mProgressDialog = ProgressDialog.show(
YouTube.this.getApplicationContext(), "",
YouTube.this.getString(R.string.loading), true);
// start time consuming background process here
}
}, 1000); // starting it in 1 second
}
Я использую Android версии 2.1 с API Level 7. Я столкнулся с этой (или похожей) проблемой и решил с помощью этого:
Dialog dialog = new Dialog(this);
вместо этого:
Dialog dialog = new Dialog(getApplicationContext());
Надеюсь это поможет:)
У меня работало переодевание
builder = new AlertDialog.Builder(getApplicationContext());
в
builder = new AlertDialog.Builder(ThisActivityClassName.this);
Странно то, что первый из них можно найти в учебнике Google, и люди получают сообщение об ошибке..
Я не думаю, что это проблема времени вокруг нулевого контекста приложения
Попробуйте расширить приложение внутри вашего приложения (или просто используйте его, если у вас уже есть)
public class MyApp extends Application
Сделайте экземпляр доступным как частный синглтон. Это никогда не ноль
private static MyApp appInstance;
Сделайте статический помощник в MyApp (который будет использовать синглтон)
public static void showProgressDialog( CharSequence title, CharSequence message )
{
prog = ProgressDialog.show(appInstance, title, message, true); // Never Do This!
}
БУМ!!
Также, посмотрите ответ инженера Android здесь: WindowManager $ BadTokenException
Одной из причин этой ошибки может быть попытка отобразить окно / диалог приложения через Контекст, который не является Деятельностью.
Теперь, я согласен, не имеет смысла, что метод принимает параметр Context вместо Activity..
Прочитав вышеупомянутые ответы, я обнаружил, что в моей ситуации следующее решило проблему.
Это выкинуло ошибку
myButton.setOnClickListener(new OnClickListener(){
public void onClick(View v) {
MyDialogue dialog = new MyDialogue(getApplicationContext());
dialog.show();
}
});
Основываясь на предыдущих ответах, в которых предполагалось, что контекст был неправильным, я изменил метод getApplicationContext(), чтобы получить контекст из переданного представления для метода onClick кнопок.
myButton.setOnClickListener(new OnClickListener(){
public void onClick(View v) {
MyDialogue dialog = new MyDialogue(v.getContext());
dialog.show();
}
});
Я не полностью понимаю работу Java, поэтому могу ошибаться, но я предполагаю, что для моей конкретной ситуации причина могла быть связана с тем фактом, что приведенный выше фрагмент был определен в классе Abstract Activity; унаследованный и используемый многими Деятельностями, возможно это способствовало тому, что getApplicationContext() не возвращает действительный контекст?? (Просто предположение).
Я создаю вид карты с подробными наложениями. Я создавал свой оверлейный элемент, как это из моей mapActivity:
OCItemizedOverlay currentLocationOverlay = new OCItemizedOverlay(pin,getApplicationContext);
Я обнаружил, что получаю исключение "android.view.WindowManager$BadTokenException: Невозможно добавить окно - маркер null не для приложения", когда мой метод itemizedoverlay запускается по методу onTap (когда местоположение отображается в mapview).
Я обнаружил, что если я просто передал "this" вместо "getApplicationContext()" моему конструктору, проблема исчезнет. Это, кажется, подтверждает заключение иностранца. странно.
Для действий, показанных в TabActivities, используйте getParent()
final AlertDialog.Builder builder = new AlertDialog.Builder(getParent());
вместо
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
Для Android 2.2
Используйте этот код:
//activity is an instance of a class which extends android.app.Activity
Dialog dialog = new Dialog(activity);
вместо этого кода:
// this code produces an ERROR:
//android.view.WindowManager$BadTokenException:
//Unable to add window -- token null is not for an application
Context mContext = activity.getApplicationContext();
Dialog dialog = new Dialog(mContext);
Примечание: мой пользовательский диалог создается за пределами activity.onCreateDialog(int dialogId)
метод.
Пытаться -
AlertDialog.Builder builder = new AlertDialog.Builder(getParent());
(Для будущих ссылок)
Я думаю, что это потому, что есть различия в контексте приложения и контексте деятельности, как описано здесь: http://www.doubleencore.com/2013/06/context/
Это означает, что мы не можем показать диалог, используя контекст приложения. Вот и все.
Была похожая проблема с (совместимостью) фрагментов, в которых используется getActivity()
в ProgressDialog.show()
вылетает это. Я бы согласился, что это из-за времени.
Возможное исправление:
mContext = getApplicationContext();
if (mContext != null) {
mProgressDialog = ProgressDialog.show(mContext, "", getString(R.string.loading), true);
}
Вместо того, чтобы использовать
mProgressDialog = ProgressDialog.show(getApplicationContext(), "", getString(R.string.loading), true);
Поместите mContext как можно раньше, чтобы у него было больше времени для захвата контекста. До сих пор нет гарантии, что это будет работать, это только снижает вероятность сбоя. Если это все еще не работает, вам придется прибегнуть к взлому таймера (что может вызвать другие проблемы с синхронизацией, такие как закрытие диалога позже).
Конечно, если вы можете использовать this
или же ActivityName.this
стабильнее, потому что this
уже указывает на что-то. Но в некоторых случаях, например с некоторыми архитектурами фрагментов, это не вариант.
Для использования диалогов внутри действий сделайте это следующим образом:
private Context mContext;
private AlertDialog.Builder mBuilder;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mContext = this;
//using mContext here refering to activity context
mBuilder = new AlertDialog.Builder(mContext);
//...
//rest of the code
//...
}
Для использования диалогов внутри фрагментов, сделайте это так:
private Context mContext;
private AlertDialog.Builder mBuilder;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View mRootView = inflater.inflate(R.layout.fragment_layout, container, false);
mContext = getActivity();
//using mContext here refering to fragment's hosting activity context
mBuilder = new AlertDialog.Builder(mContext);
//...
//rest of the code
//...
return mRootView;
}
Вот и все ^_^
Это распространенная проблема. использование this
вместо getApplicationContext()
Это должно решить вашу проблему
Чтобы обойти это, я создал базовый класс для всех моих действий, где я храню глобальные данные. В первом упражнении я сохранил контекст в переменной базового класса следующим образом:
Базовый класс
public static Context myucontext;
Первое действие, полученное из базового класса
mycontext = this
Затем я использую mycontext вместо getApplicationContext при создании диалогов.
AlertDialog alertDialog = new AlertDialog.Builder(mycontext).create();
Если вы вызываете ProgressDialog.show() во фрагменте, приведение mContext к Activity сработало для меня.
ProgressDialog pd = new ProgressDialog((Activity) mContext);
Диалог всегда создается и отображается как часть действия. Вам нужно передать контекст Activity вместо контекста Application.
Если у вас есть проблема с groupActivity, не используйте это. PARENT - это статический элемент из Parent ActivityGroup.
final AlertDialog.Builder builder = new AlertDialog.Builder(GroupActivityParent.PARENT);
вместо
final AlertDialog.Builder builder = new AlertDialog.Builder(getParent());
Я реализовал Alert Dialog для выдачи исключений в текущем представлении активностей.
AlertDialog.Builder builder = new AlertDialog.Builder(context);
Учитывая то же самое Исключение Окна. Я пишу код для предупреждения из onCreate(). Так просто я использовал context = this;
после setContentView()
заявление в onCreate()
method.Taken переменная контекста как глобальная Context context;
Пример кода
static Context context;
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.network);
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
context = this;
.......
Пример метода оповещения
private void alertException(String execMsg){
Log.i(TAG,"in alertException()..."+context);
Log.e(TAG,"Exception :"+execMsg);
AlertDialog.Builder builder = new AlertDialog.Builder(context);
.......
Он отлично работает для меня. На самом деле я искал эту ошибку в Stackru, я нашел этот запрос. После прочтения всех ответов этого поста, я попробовал этот способ, чтобы он работал. Я думал, что это простое решение для преодоления исключения.
Спасибо, Раджендар