ActivityネストFragmentアプリケーションは長い間バックグラウンドに置かれ、システムに回収され、crashが現れた.

19170 ワード

メインインタフェースを使用したものが多くメモリを消費するため、プログラムがバックグラウンドに入った後、長時間操作せずにシステムに回収され、再びプログラムに入った後、プログラムがクラッシュした.エラーログ:
Parcel android.os.Parcel@42209460: Unmarshalling unknown type code
 2131165303 at offset 3748
         at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2080)
         at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2105)
         at android.app.ActivityThread.access$600(ActivityThread.java:136)
         at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1201)
         at android.os.Handler.dispatchMessage(Handler.java:99)
         at android.os.Looper.loop(Looper.java:137)
         at android.app.ActivityThread.main(ActivityThread.java:4876)
         at java.lang.reflect.Method.invokeNative(Native Method)
         at java.lang.reflect.Method.invoke(Method.java:511)
         at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:804)
         at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:571)
         at com.kdgdev.xtension.core.XtensionMain.main(XtensionMain.java:91)
         at dalvik.system.NativeStart.main(Native Method)

ログを見ると、アプリケーションがクラッシュしてパケット解除されたオフセットが異常になります.
エラー位置への位置決め
FragmentActivity super.onCreate(savedInstanceState);       

エラーを報告したFragmentActivityのソースコードを確認します.
FragmentActivity
protected void onSaveInstanceState(Bundle outState)
{
super.onSaveInstanceState(outState);
Parcelable p = this.mFragments.saveAllState();
if (p != null)
outState.putParcelable("android:support:fragments", p);
}
static final String FRAGMENTS_TAG = "android:support:fragments";
/**
     * Perform initialization of all fragments and loaders.
     */
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        mFragments.attachActivity(this, mContainer, null);
        // Old versions of the platform didn't do this!
        if (getLayoutInflater().getFactory() == null) {
            getLayoutInflater().setFactory(this);
        }
        
        super.onCreate(savedInstanceState);
        
        NonConfigurationInstances nc = (NonConfigurationInstances)
                getLastNonConfigurationInstance();
        if (nc != null) {
            mAllLoaderManagers = nc.loaders;
        }
        if (savedInstanceState != null) {
            Parcelable p = savedInstanceState.getParcelable(FRAGMENTS_TAG);
            mFragments.restoreAllState(p, nc != null ? nc.fragments : null);
        }
        mFragments.dispatchCreate();
    }

クラッシュ時にView状態を保存する処理を行い、クラッシュ回復後にgetParcelable(FRAGMENTS_TAG)を解包する
リカバリView状態でシフトエラー(オブジェクトがParcelableでない場合、
writeToParcelとcreateFromParcelが対応していないなど)
マルチポジショニングは、AutoScrollTextViewの実装に問題があることを発見し、クラッシュの原因となった場合、Viewのステータスが誤って保存されます.エラーが発生した場所を表示します.
public class AutoScrollTextView extends TextView implements OnClickListener {
    public final static String TAG = AutoScrollTextView.class.getSimpleName();
   
    private float textLength = 0f;//    
    private float viewWidth = 0f;
    private float step = 0f;//      
    private float y = 0f;//      
    private float temp_view_plus_text_length = 0.0f;//         
    private float temp_view_plus_two_text_length = 0.0f;//         
    public boolean isStarting = false;//      
    private Paint paint = null;//    
    private String text = "";//    
   
    public AutoScrollTextView(Context context) {
        super(context);
        initView();
    }
    public AutoScrollTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }
    public AutoScrollTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        initView();
    }
   
   
    private void initView()
    {
        setOnClickListener(this);
    }
   
   
    public void init(WindowManager windowManager)
    {
        paint = getPaint();
        text = getText().toString();
        textLength = paint.measureText(text);
        viewWidth = getWidth();
        if(viewWidth == 0)
        {
            if(windowManager != null)
            {
                Display display = windowManager.getDefaultDisplay();
                viewWidth = display.getWidth();
            }
        }
        step = textLength;
        temp_view_plus_text_length = viewWidth + textLength;
        temp_view_plus_two_text_length = viewWidth + textLength * 2;
        y = getTextSize() + getPaddingTop();
    }
   
    @Override
    public Parcelable onSaveInstanceState()
    {
        Parcelable superState = super.onSaveInstanceState();
        SavedState ss = new SavedState(superState);
       
        ss.step = step;
        ss.isStarting = isStarting;
       
        return ss;
       
    }
   
    @Override
    public void onRestoreInstanceState(Parcelable state)
    {
        if (!(state instanceof SavedState)) {
            super.onRestoreInstanceState(state);
            return;
        }
        SavedState ss = (SavedState)state;
        super.onRestoreInstanceState(ss.getSuperState());
       
        step = ss.step;
        isStarting = ss.isStarting;
    }
   
    public static class SavedState extends BaseSavedState {
        public boolean isStarting = false;
        public float step = 0.0f;
        SavedState(Parcelable superState) {
            super(superState);
        }
        @Override
        public void writeToParcel(Parcel out, int flags) {
            super.writeToParcel(out, flags);
            out.writeBooleanArray(new boolean[]{isStarting});
            out.writeFloat(step);
        }

        public static final Parcelable.Creator CREATOR
                = new Parcelable.Creator() {
           
            public SavedState[] newArray(int size) {
                return new SavedState[size];
            }
            @Override
            public SavedState createFromParcel(Parcel in) {
                return new SavedState(in);
            }
        };
        private SavedState(Parcel in) {
            super(in);
            boolean[] b = null;
            in.readBooleanArray(b);
            if(b != null && b.length > 0)
                isStarting = b[0];
            step = in.readFloat();
        }
    }
   
    public void startScroll()
    {
        isStarting = true;
        invalidate();
    }
   
   
    public void stopScroll()
    {
        isStarting = false;
        invalidate();
    }
   
    @Override
    public void onDraw(Canvas canvas) {
        canvas.drawText(text, temp_view_plus_text_length - step, y, paint);
        if(!isStarting)
        {
            return;
        }
        step += 0.5;//0.5       。
        if(step > temp_view_plus_two_text_length)
            step = textLength;
        invalidate();
    }
    @Override
    public void onClick(View v) {
        if(isStarting)
            stopScroll();
        else
            startScroll();
       
    }
}

走馬灯ではTextViewでカスタム実装SavedStateによる解包異常を書き換え,SavedStateの実装を注釈すると正常に戻る.ほとんどのアプリケーションはこのようにして、破棄して再createしたので、間違ったところを判断すればいいので、もう一つの解決策はFragmentActivityのonCreateで判断することです.
if (savedInstanceState != null) {
     savedInstanceState.putParcelable("android:support:fragments", null);
}
super.onCreate(savedInstanceState);
 
   如果是Activity,则改成如下: 
  
if (savedInstanceState != null) {
     savedInstanceState.putParcelable("android:fragments", null);
}
super.onCreate(savedInstanceState);