【问题标题】:java.lang.IllegalStateException: Activity has been destroyed Exceptionjava.lang.IllegalStateException: Activity 已被销毁异常
【发布时间】:2013-12-09 14:58:06
【问题描述】:

我正在实现一个播放视频的应用程序。有 2 个活动:MainMenuActivity(纵向模式)和VideoPlayerActivity(可旋转)。 MainMenuActivity 有 HomeFragmentLT 片段。在HomeFragmentLT 片段中有一个 ViewPager。这是我的HomeFragmentLT 和 ViewPager(HomePager):

public class HomeFragmentLT extends Fragment implements Observer
{
    public static String TAG = "HomeFragmentLT";
    private ViewGroup mLayout;
    private View mHeader;
    private HomeAdapter mHomeAdapter;       
@Override
public void onCreate(Bundle savedInstanceState) {
    // TODO Auto-generated method stub
    super.onCreate(savedInstanceState);
    Log.e(TAG, "onCreate");
}

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
{
    if (mLayout == null)        
        mLayout = (ViewGroup) inflater.inflate(R.layout.home_listview, null);       
    else        
        ((ViewGroup) mLayout.getParent()).removeView(mLayout);      
    initViews();        
    return mLayout;
}

public HomeFragmentLT(){}

private void initViews()
{       
    mPager = (ViewPager) mHeader.findViewById(R.id.home_pager);
    .......
    .......
}

@Override
public void onStart()
{
    super.onStart();
    if (!mDataReceived)
        getData();      
    Log.e(TAG, "onStart");
}

@Override
public void onStop()
{
    DataHelper.getInstance().deleteObserver(this);
    super.onStop();
    Log.e(TAG, "onStop");
}

@Override
public void onDestroy() {
    // TODO Auto-generated method stub
    super.onDestroy();
    Log.e(TAG, "onDestroy");
}

private void getData()
{
    if (mError)
    {
        UIUtils.toggleError(mLayout, false, null);
        mListView.setOnTouchListener(null);
        UIUtils.toggleLoading(mLayout, true);
    }
    DataHelper.getInstance().addObserver(this);
    DataHelper.getInstance().getHomeData();
}

private void setData()
{
    mHomeAdapter = new HomeAdapter(getActivity(), mPager, mHomeData.getHotPrograms());
    mPager.setAdapter(mHomeAdapter);
    mIndicator.setViewPager(mPager);

    .............
}

@Override
public void update(Observable observable, Object data)
{
    if (data == null)
        return;
    Result result = (Result) data;
    if (Method.GET_HOME_DATA.equals(result.getParams().getMethod()))
    {
        if (result.getError() == null)
        {
            mHomeData = (HomeObj) result.getData();
            setData();
            mDataReceived = true;
            mError = false;
            DataHelper.getInstance().deleteObserver(this);
        }           
    }
}
......................
public static class HomeAdapter extends PagerAdapter
{
    private List<Program> mPrograms;
    private View[] mViews;
    private Context mContext;

    public HomeAdapter(Context context, ViewPager viewpager, List<Program> programs){...........}

    @Override
    public int getCount()
    {
        return mPrograms.size();
    }

    @Override
    public boolean isViewFromObject(View view, Object object)
    {
        return view == object;
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position)
    {
        View view;
        if ((view = mViews[position]) != null)
        {
            ImageView img = (ImageView) view.findViewById(R.id.img);
            if (img.getDrawable() == null)
            {
                UIUtils.toggleLoading(view, true);
                ImageManager.getInstance().push(mPrograms.get(position).getThumbnail(), img, mLoadingListener);
            }
            return view;
        }

        view = LayoutInflater.from(mContext).inflate(R.layout.home_program_item_top, null);
        view.setTag(mPrograms.get(position).getId());
        view.setLayoutParams(mParams);
        if (Constants.MAY_CUI_DETECTED)
            ImageManager.getInstance().push(mPrograms.get(position).getThumbnail(),
                    (ImageView) view.findViewById(R.id.img), mParams.width, mParams.height, mLoadingListener);
        else
            ImageManager.getInstance().push(mPrograms.get(position).getThumbnail(),
                    (ImageView) view.findViewById(R.id.img), mLoadingListener);

        view.setOnClickListener(mOnItemClick);
        container.addView(view);
        return mViews[position] = view;
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object){}
}   
}

MainMenuActivity 中,我在onCreate() 中添加HomeFragmentLT

getSupportFragmentManager().beginTransaction().add(R.id.dummy, mHomeFragment).commitAllowingStateLoss();
getSupportFragmentManager().executePendingTransactions();

一切正常。但是当我将手机置于横向时,然后启动VideoPlayerActivity(意味着横向模式下的VideoPlayerActivity)播放来自MainMenuActivity(纵向)的视频并立即按下返回按钮。它得到了例外。这是我的日志:

12-09 18:30:09.102: E/MainMenuActivity(23287): onCreate()
12-09 18:30:09.332: E/MainMenuActivity(23287): onStart()
12-09 18:30:09.432: E/MainMenuActivity(23287): onResume()
12-09 18:30:09.703: E/HomeFragmentLT(23287): onCreate()
12-09 18:30:09.703: E/HomeFragmentLT(23287): onCreateView()
...........
12-09 18:30:14.487: E/MainMenuActivity(23287): onPause()
12-09 18:30:14.788: E/VideoPlayerActivity(23287): onCreate()
12-09 18:30:14.788: E/VideoPlayerActivity(23287): onStart()
...........
12-09 18:30:15.418: E/VideoPlayerActivity(23287): onPause()
12-09 18:30:15.588: E/HomeFragmentLT(23287): onStop()
12-09 18:30:15.639: E/MainMenuActivity(23287): onDestroy()
12-09 18:30:15.659: E/MainMenuActivity(23287): onCreate()
12-09 18:30:15.659: E/HomeFragmentLT(23287): onCreate()
12-09 18:30:15.719: E/MainMenuActivity(23287): onStart()
12-09 18:30:15.789: E/HomeFragmentLT(23287): onCreateView()
12-09 18:30:15.979: E/MainMenuActivity(23287): onResume()
12-09 18:30:16.169: E/MainMenuActivity(23287): onPause()
12-09 18:30:16.169: E/HomeFragmentLT(23287): onStop()
12-09 18:30:16.179: E/MainMenuActivity(23287): onDestroy()
12-09 18:30:16.199: E/MainMenuActivity(23287): onCreate()
12-09 18:30:16.199: E/HomeFragmentLT(23287): onCreate()
12-09 18:30:16.239: E/MainMenuActivity(23287): onStart()
12-09 18:30:16.259: E/HomeFragmentLT(23287): onCreateView()
12-09 18:30:16.379: E/MainMenuActivity(23287): onResume()
12-09 18:30:16.469: E/AndroidRuntime(23287): FATAL EXCEPTION: main
12-09 18:30:16.469: E/AndroidRuntime(23287): java.lang.IllegalStateException: Activity has been destroyed
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.support.v4.app.FragmentManagerImpl.enqueueAction(FragmentManager.java:1358)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.support.v4.app.BackStackRecord.commitInternal(BackStackRecord.java:595)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.support.v4.app.BackStackRecord.commitAllowingStateLoss(BackStackRecord.java:578)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at com.vng.zingtv.activity.MainMenuActivity$3.run(MainMenuActivity.java:375)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.os.Handler.handleCallback(Handler.java:615)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.os.Handler.dispatchMessage(Handler.java:92)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.os.Looper.loop(Looper.java:137)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at android.app.ActivityThread.main(ActivityThread.java:4744)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at java.lang.reflect.Method.invokeNative(Native Method)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at java.lang.reflect.Method.invoke(Method.java:511)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:786)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
12-09 18:30:16.469: E/AndroidRuntime(23287):    at dalvik.system.NativeStart.main(Native Method)

有 2 个问题需要您的帮助。
1. 为什么MainMenuActivity被销毁又重新创建? (我的 LogCat)
2. 我的HomeFragmentLTfragment 有什么问题导致IllegalStateException

提前致谢。

编辑
我希望我的 VideoPlayerActivity 根据设备方向旋转。所以在manifyt.xml中

<activity
     android:name=".activity.VideoPlayerActivity"
     android:screenOrientation="portrait" />

在 VideoPlayerActivity 的 onCreate() 中:

public void onCreate(Bundle savedInstanceState){
    super.onCreate(savedInstanceState);
        initViews();
        ......

    setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR);
}

【问题讨论】:

  • 你的意思是方向改变?
  • 没有。我的 VideoPlayerActivity 可以根据设备方向旋转。如果我以纵向模式启动 VideoPlayerActivity。没关系。但是如果我从 MainMenuActivity 以横向模式启动它并立即按 Back 按钮。它会导致 MainMenuActivity 被破坏并重新创建(我的日志猫)并获得异常。

标签: android android-fragments android-viewpager


【解决方案1】:

旋转设备,意味着系统必须加载新资源。例如。您可能在某些需要加载的 layout-land 文件夹中有不同的资源。为了能够加载这些新资源,Activity 被销毁并使用新的 drawables/layouts/values 等重新创建。Android 具有用于此类更改的特殊机制,称为configChanges。这包括设备旋转,还包括更改手机语言等。

为避免您的活动被破坏,您可以在AndroidManifest.xml 中声明,您将通过设置此标志自行处理方向更改:

<activity
    android:name=".VideoPlayerActivity"
    android:configChanges="orientation|screenSize" />

在您的活动中,添加以下代码:

@Override
public void onConfigurationChanged(Configuration newConfig) {
    super.onConfigurationChanged(newConfig);

    // Checks the orientation of the screen
    if (newConfig.orientation == Configuration.ORIENTATION_LANDSCAPE) {
        Toast.makeText(this, "landscape", Toast.LENGTH_SHORT).show();
    } else if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT){
        Toast.makeText(this, "portrait", Toast.LENGTH_SHORT).show();
    }
}

欲了解更多信息,请参阅this link to the developer documentation

【讨论】:

  • 感谢您的宝贵时间。但我希望我的 VideoPlayerActivity 根据设备方向旋转。如果我启动 VideoPlayerActivity,然后稍后按返回按钮。没关系。但是,如果我立即按下后退按钮(在统计 VideoPlayerActivity 时几乎相同),它会导致异常。这会导致我的 MainMenuActivity(启动 VideoPlayerActivity)。我真的不知道为什么
【解决方案2】:

我不知道为什么 MainMenuActivity 被破坏并重新创建了 2 次,因为 MainMenuActivity 被旋转了。(我没有看到 MainMenuActivity 被旋转,只是从 logcat 中猜测)。为了避免这种情况,我在 manifiest.xml 中添加了android:configChanges="orientation|keyboardHidden|screenSize"

<activity
      android:name=".activity.MainMenuActivity"
      android:screenOrientation="portrait"
      android:configChanges="orientation|keyboardHidden|screenSize"/>

现在我可以在 MainMenuActivity 中看到onConfigurationChanged() 的日志,避免 MainMenuActivity 被销毁并重新创建。 但我仍然不知道为什么它被破坏了。 现在是日志猫:

12-10 00:00:16.865: E/VideoPlayerActivity(20105): onPause()
12-10 00:00:16.885: E/MainMenuActivity(20105): onConfigurationChanged LANDSCAPE
12-10 00:00:16.885: E/MainMenuActivity(20105): onResume()
12-10 00:00:17.296: E/MainMenuActivity(20105): onConfigurationChanged PORTRAIT
12-10 00:00:17.686: E/VideoPlayerActivity(20105): onStop()
12-10 00:00:17.686: E/VideoPlayerActivity(20105): onDestroy()

【讨论】:

  • 您最好以正确的方式解决此问题,然后在需要时禁用方向。因为在某些情况下,系统会重新创建 Activity,您可能会遇到同样的崩溃。
  • 你知道为什么 MainMenuActivity 会被销毁并重新创建吗? (甚至 2 次)。如果我轻松按下后退按钮。它工作正常。否则我立即按下返回按钮,它崩溃了。 MainMenuActivity 被重新创建。
  • 如果您不自己处理 configChanges,它将在轮换后重新创建。你得到崩溃是因为你做错了什么。您没有提供足够的代码来查看到底出了什么问题。
  • 但在 manifiest.xml 中,我将 android:screenOrientation="portrait" 设置为 MainMenuActivity。为什么要重新创建?如果我在启动 VideoPlayerActivity 大约 1 秒后按返回按钮。它工作正常。 MainMenuActivity 没有重新创建。否则,我在启动 VideoPlayerActivity 后立即按返回按钮。它被重新创建。
  • 您所描述的案例听起来像是被破坏了,因为您在播放视频时缺少资源。无论如何,这对Android来说是正常的。当您按下返回时,它会重新创建。为避免崩溃,您需要遵循以下规则:不要从 Activity 之外引用 Activity 或其视图。尝试仅在短时间内且仅在必要时引用 Fragment 中的 Activity,主要是在 onCreate 或 onCreateView 中。如果 AsyncTasks 已启动,请在 Activity onStop 中取消它们。进入 Fragment 的 getActivity() 后总是检查 null。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-06-23
  • 2016-06-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多