2013-07-24 3 views
1

Недавно я получил авария в своей программе. Используя logCat, я определил, что авария произошла из-за кода в моем onRestoreInstanceState() в основном действии, когда мое приложение возобновляется. Logcat файл:Bundle null in onRestoreInstanceState?

07-23 16:27:01.927 I/ActivityManager( 390): START u0 {act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.myapp/.MainActivity bnds=[1184,809][1376,1001]} from pid 30666 

07-23 16:27:01.997 I/ActivityManager( 390): Start proc com.myapp for activity com.myapp/.MainActivity: pid=31007 uid=10070 gids={50070, 1028} 

07-23 16:27:02.137 E/AndroidRuntime(31007): FATAL EXCEPTION: main 


07-23 16:27:02.137 E/AndroidRuntime(31007): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.myapp/com.myapp.MainActivity}: java.lang.ClassCastException: android.os.Parcelable[] cannot be cast to com.myapp.Panel[] 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2180) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2230) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread.access$600(ActivityThread.java:141) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1234) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.os.Handler.dispatchMessage(Handler.java:99) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.os.Looper.loop(Looper.java:137) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread.main(ActivityThread.java:5041) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at java.lang.reflect.Method.invokeNative(Native Method) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at java.lang.reflect.Method.invoke(Method.java:511) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at dalvik.system.NativeStart.main(Native Method) 

07-23 16:27:02.137 E/AndroidRuntime(31007): Caused by: java.lang.ClassCastException: android.os.Parcelable[] cannot be cast to com.myapp.Panel[] 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at com.myapp.MainActivity.onRestoreInstanceState(MainActivity.java:177) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.Activity.performRestoreInstanceState(Activity.java:910) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.Instrumentation.callActivityOnRestoreInstanceState(Instrumentation.java:1131) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2158) 

07-23 16:27:02.137 E/AndroidRuntime(31007):  ... 11 more 

07-23 16:27:02.167 W/ActivityManager( 390): Force finishing activity com.myapp/.MainActivity 

Катастрофа происходит на линии mPanels = (Panel[]) inState.getParcelableArray("panelParcel"); в onRestoreInstanceState ниже методом.

Эта методология прочно работала для меня до этого сбоя, и я пытаюсь понять, почему у объекта inState нет массива объектов (Панелей), которые я сохранил в нем. Я смущен, потому что кажется, что inState не включает панель Panel, которую я сохранил.

Я вижу из this question я обычно должен использовать onCreate вместо onRestoreInstanceState, но я хотел бы понять, почему это не работал бы в редком случае, возможно, я не в полной мере понять что-то о жизненном цикле активность?

код как для сохранения и восстановления экземпляра Состояние:

@Override 
public void onSaveInstanceState(@NotNull Bundle outState) 
{ 
    Log.d("panelCreation", "onSaveInstanceState Called"); 
    super.onSaveInstanceState(outState); 
    // save the current panel state 
    outState.putParcelableArray("panelParcel", mPanels); 
} 

@Override 
public void onRestoreInstanceState(Bundle inState) 
{ 
    Log.d("panelCreation", "onsState Called"); 
    super.onRestoreInstanceState(inState); 

    // Note getParcelable returns a *new* array, so we must setup the drawer listener after this 
    mPanels = (Panel[]) inState.getParcelableArray("panelParcel"); 
    // set up the drawer's list view with items and click listener 
    mDrawerList.setAdapter(new PanelArrayAdapter(this, 
      R.layout.drawer_list_item, mPanels)); 
    refreshDrawerListChecked(); 

    // Forces the Adapter to redraw the view to ensure color stylings are applied 
    ((PanelArrayAdapter)mDrawerList.getAdapter()).notifyDataSetChanged(); 
} 

ответ

3

Вы можите литой массив Parcelable на массив Panel.

См casting Object array to Integer array error

+0

Имейте в виду, что это работает подавляющее большинство времени - я просто получаю «удачу» в моем кастинге, когда он работает? – CrimsonX

+0

Я думаю, что вы можете быть правы - будет исследовать, это сообщение выглядит весьма полезным и похоже на мою проблему: http://stackoverflow.com/questions/10071502/read-writing-arrays-of-parcelable-objects – CrimsonX

+1

As i понимаете, в общем, вы не можете использовать 'SuperType []' для 'DerivedType []', если компилятор/интерпретатор не знает, что 'SuperType []' действительно 'DerivedType []'. В вашем случае интерпретатор не знает, что десериализованный массив является массивом 'Panel'. Поэтому вам нужно использовать каждый элемент массива или использовать 'Arrays.copyOf (a, a.length, Panel []. Class);' где 'a' является' Parcelable [] a = inState.getParcelableArray ("panelParcel") ' , – JustAnotherCoder

0

Посмотрите на этот пост для получения дополнительной информации onSaveInstanceState() and onRestoreInstanceState()

«onRestoreInstanceState() вызывается только при воссоздании активности после того, как он был убит OS»

и из документации:

Mo st будет просто использовать onCreate (Bundle), чтобы восстановить их состояние , но иногда это удобно делать после завершения инициализации или разрешить подклассам решать, использовать ли для реализации по умолчанию. Реализация по умолчанию этого метода выполняет восстановление любого состояния представления, которое ранее было , замерзаемое onSaveInstanceState (Bundle).

Кроме того, этот ответ может быть полезным https://stackoverflow.com/a/2909211/1241244

+0

Я согласен с вами - я даже упомянул в моем вопросе, что я собираюсь начать использовать 'onCreate' вместо' onRestoreInstanceState', но это все еще оленья кожа ответить на мой вопрос. – CrimsonX

Смежные вопросы