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.

http://developer.android.com/guide/topics/ui/dialogs.html

Если у вас есть проблема с 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, я нашел этот запрос. После прочтения всех ответов этого поста, я попробовал этот способ, чтобы он работал. Я думал, что это простое решение для преодоления исключения.

Спасибо, Раджендар

Другие вопросы по тегам