2012-06-12 2 views
7

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

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

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

+0

Я думаю, что ваш лучший снимок будет ухватить исходного кода и шаг за шагом через него с помощью отладчика. Очень может быть ошибкой в ​​Android. –

+0

Я не сделал никакого прогресса в этой проблеме. Я прошел через код, но не достаточно хорошо знаком с жизненным циклом фрагмента/активности за кулисами, чтобы узнать, что происходит или что может быть проблемой. Сейчас у меня есть обходной путь, который довольно «взломан», но я двигаюсь дальше. Если кто-нибудь придет с ответом, дайте мне знать. Благодаря! – Dave

+0

У меня такая же проблема. 'onCreateOptionMenu' и' onPrepareOptionsMenu' вызывается дважды, а во второй раз они перезапускают состояние меню. Еще не нашли решение :( –

ответ

5

Я думаю, что нашел ответ на эту проблему.

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

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

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

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

setRetainInstance(true); 

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

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

+3

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

2

@Gerardo Contijoch ответ вводит в заблуждение, за один факт:

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

К setRetainInstance(true) вы сообщаете Android, чтобы не разрушать фрагмент. Это может оказаться полезным в случае фрагмента UI-less, не содержащего контекста Activity (, полезного для координации AsyncTasks и некоторых других подобных сервисов).

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

0

Я просто очищается меню перед накачиванием и работает

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

}