Android onCreateOptionsMenu вызывается дважды при восстановлении состояния

Вот простое приложение для Android, которое я создал, чтобы продемонстрировать мою проблему:

public class OptionMenuTest extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.d("test", "create activity");
        setContentView(R.layout.options_layout);
        if(getFragmentManager().findFragmentByTag("frag") == null) {
            getFragmentManager().beginTransaction().add(R.id.option_fragment_container, new OptionMenuFragment(), "frag").commit(); 
        }

    }

    @Override
    protected void onSaveInstanceState(Bundle outState) {
        Log.d("test", "saving Activity state");
        super.onSaveInstanceState(outState);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        Log.d("test", "create Activity options menu");
        menu.add("activity");
        return true;
    }
}

Фрагмент:

public class OptionMenuFragment extends Fragment {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.d("test", "create fragment");
        setHasOptionsMenu(true);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        TextView tv = new TextView(getActivity());
        tv.setText("Hello world");
        Log.d("test", "create fragment view");
        return tv;
    }

    @Override
    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        menu.add("fragment");
        Log.d("test", "create fragment options menu");
    }
}

Layout - это просто LinearLayout, чтобы выгрузить фрагмент в:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/option_fragment_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    >
</LinearLayout>

Очень просто, верно? Когда я запускаю его, я получаю следующий вывод:

06-12 15:42:51.415: D/test(957): create activity
06-12 15:42:51.446: D/test(957): create fragment
06-12 15:42:51.446: D/test(957): create fragment view
06-12 15:42:51.446: D/test(957): create Activity options menu
06-12 15:42:51.446: D/test(957): create fragment options menu

Теперь, когда я поворачиваю телефон, я получаю странное поведение:

06-12 15:43:11.251: D/test(957): saving Activity state
06-12 15:43:11.290: D/test(957): create fragment
06-12 15:43:11.290: D/test(957): create activity
06-12 15:43:11.306: D/test(957): create fragment view
06-12 15:43:11.306: D/test(957): create Activity options menu
06-12 15:43:11.306: D/test(957): create fragment options menu
06-12 15:43:11.306: D/test(957): create Activity options menu
06-12 15:43:11.306: D/test(957): create fragment options menu

Почему операция onCreateOptionMenu и фрагмент onCreateOptionsMenu вызываются дважды? Если я удаляю меню параметров из фрагмента, я получаю 1 вызов к действию onCreateOptionsMenu, как и ожидалось:

06-12 15:50:03.610: D/test(1076): create fragment
06-12 15:50:03.610: D/test(1076): create fragment view
06-12 15:50:03.813: D/test(1076): create Activity options menu
06-12 15:50:08.392: D/test(1076): saving Activity state // <-- rotate happens here
06-12 15:50:08.446: D/test(1076): create fragment
06-12 15:50:08.446: D/test(1076): create activity
06-12 15:50:08.462: D/test(1076): create fragment view
06-12 15:50:08.470: D/test(1076): create Activity options menu

Я не понимаю этого, и никто другой, похоже, не сталкивался с этой проблемой. Реальная проблема заключается в том, что мой SearchView не может восстановить свое состояние при изменении конфигурации (поворот телефона), поскольку onCreateOptionMenu вызывается дважды. В первый раз он, кажется, имеет его состояние, но во второй раз его очищают и сбрасывают. Я не могу понять, что я делаю неправильно.

Заранее спасибо.

 Dave14 июн. 2012 г., 19:04
Я не продвинулся в решении этой проблемы. Я прошелся по коду, но недостаточно знаком с закулисным жизненным циклом / деятельностью, чтобы понять, что происходит или в чем может быть проблема. У меня прямо сейчас есть обходной путь, который довольно "хакерский" но я иду дальше. Если кто-нибудь придет с ответом, дайте мне знать. Спасибо!
 Gerardo Contijoch08 июл. 2012 г., 01:10
У меня точно такая же проблема.onCreateOptionMenu а такжеonPrepareOptionsMenu Вам звонят дважды, и во второй раз они сбрасывают состояние меню. Еще не нашли решение :(
 Timo Ohr12 июн. 2012 г., 22:37
Я думаю, что ваш лучший способ - взять исходный код и пройти его с помощью отладчика. Вполне может быть ошибкой в Android.

Ответы на вопрос(3)

за исключением одного факта:

В вашем примере обаActivity а такжеFragment разрушаться при вращении и создаваться заново. ПоэтомуonCreateOptionsMenu() называется дважды. Это правильное и ожидаемое поведение.

ОтsetRetainInstance(true) вы говорите Android не уничтожать фрагмент. Это может оказаться полезным в случае, когда фрагмент без пользовательского интерфейса не содержит Activity & apos; контекст (useful for AsyncTasks coordination and some other service-like stuff).

В других случаях фрагментация может привести к утечке памяти, чего следует избегать.

 @Override
 public void   onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
      menu.clear();
      inflater.inflate(R.menu.call_menu, menu);
      super.onCreateOptionsMenu(menu, inflater);

 }
Решение Вопроса

Посмотрите это:

https://stackoverflow.com/a/7225296/48468

Кажется, проблема связана с тем, что Android не уничтожает фрагмент, когда активность уничтожается (когда устройство поворачивается).

В основном я добавил:

setRetainInstance(true);

моему конструктору фрагментов и проблема решена.

Надеюсь, поможет!

 Dave09 июл. 2012 г., 15:23
Да! Спасибо огромное. это решило мою проблему. Интересно, что у действия также есть эта проблема, если фрагмент участвует в меню параметров и не вызывает setRetainInstance (true), даже если SearchView добавляется действием. Если вы удалите setHasOptionsMenu из фрагмента, действие SearchView будет работать нормально. Я думаю, потому что вновь добавленный фрагмент вызывает действия onCreateOptionsMenu для повторного вызова.

Ваш ответ на вопрос