当前位置: 首页 > 知识库问答 >
问题:

Android:活动意外销毁,空savedInstanceState

鞠源
2023-03-14

在发布本文之前,我不会列出我读过的所有参考资料,但我已经广泛阅读了有关stackoverflow的类似问题,以及所有关于活动/片段生命周期和维护状态的android开发人员文档,还没有找到解决方案。

以下是我的设想:

我有一个主要的活动(PulseActivity),它使用FragmentPagerAdapter。每个片段显示我从服务器检索的数据的单独列表。按下PulseActivity的actionbar中的action_项时,可以过滤从服务器发送的数据。为此,PulseActivity启动一个单独的活动(FilterEventActivity)。FilterEventsActivity将PulseActivity列为其父活动。按FilterEventActivity上的back按钮时,将使用null(Bundle savedInstanceState)重新创建PulseActivity(OnCreate调用)。出于测试目的,我覆盖了OnSaveInstanceState并在outState中放置假数据。请注意,从PulseActivity启动的每个活动都会发生这种情况。

我的问题是:

在这种特殊情况下,防止每个片段需要重新加载数据的正确设计范式是什么?我宁愿不使用单例模式,因为我的片段在其他活动中被重用。

相关源代码(如果需要):

public class ImpulseActivity extends FragmentActivity implements
    ActionBar.TabListener {

private MapSearchFragment mSearchFragment;
private BulletinFragment mBulletinFragment;

SectionsPagerAdapter mSectionsPagerAdapter;
ViewPager mViewPager;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.impulse_activity);

    Log.v("ImpulseActivity", "onCreate " + savedInstanceState);

    final ActionBar actionBar = getActionBar();
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);

    CategoryManager manager = CategoryManager.getManager();
    manager.setListener(this);
    manager.loadCategories();

    mSectionsPagerAdapter = new SectionsPagerAdapter(
            getSupportFragmentManager());

    mViewPager = (ViewPager) findViewById(R.id.pager);
    mViewPager.setAdapter(mSectionsPagerAdapter);
    mViewPager
            .html" target="_blank">setOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() {
                @Override
                public void onPageSelected(int position) {
                    actionBar.setSelectedNavigationItem(position);
                }
            });

    for (int i = 0; i < mSectionsPagerAdapter.getCount(); i++) {
        actionBar.addTab(actionBar.newTab()
                .setText(mSectionsPagerAdapter.getPageTitle(i))
                .setTabListener(this));
    }
}

@Override
public void onSaveInstanceState(Bundle outState) {
    super.onSaveInstanceState(outState);
    outState.putDouble("Hello", 1.02);
}


@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.impulse_activity_actions, menu);
    return super.onCreateOptionsMenu(menu);
}

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    int id = item.getItemId();
    if (id == R.id.action_create) {
        return true;
    }
    if (id == R.id.action_filter) {
        Intent intent = new Intent(this, FilterEventsActivity.class);
        startActivity(intent);
        return true;
    }
    return super.onOptionsItemSelected(item);
}

@Override
public void onTabSelected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
    int position = tab.getPosition();
    mViewPager.setCurrentItem(position, true);
}

@Override
public void onEventListClick(Event e) {
    Intent mIntent = new Intent(this, EventActivity.class);
    Bundle mBundle = new Bundle();
    mBundle.putSerializable(EventDetailsFragment.Event_Key, e);
    mIntent.putExtras(mBundle);
    startActivity(mIntent);
}

public class SectionsPagerAdapter extends FragmentPagerAdapter {

    public SectionsPagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @Override
    public Fragment getItem(int position) {

        Fragment fragment;
        switch(position) {
            case 0:
                if (mBulletinFragment != null) {
                    fragment = mBulletinFragment;
                } else {
                    fragment = new BulletinFragment();
                    mBulletinFragment = (BulletinFragment) fragment;
                }
                break;
            case 1:
                fragment = new MapSearchFragment();
                mSearchFragment = (MapSearchFragment) fragment;
                break;
            case 2:
                fragment = new MyEventsFragment();
                break;
            default:
                fragment = new BulletinFragment();
                break;
        }
        Bundle args = new Bundle();
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public int getCount() {
        // Show 3 total pages.
        return 3;
    }

    @Override
    public CharSequence getPageTitle(int position) {
        Locale l = Locale.getDefault();
        switch (position) {
            case 0:
                return "Bulletin";
            case 1:
                return "Map";
            case 2:
                return "My Events";
            default:
                return "Test";
        }
    }
}
}

AndroidManifest.xml

<activity
        android:name=".Home.ImpulseActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity
        android:name=".Event.EventActivity"
        android:label="@string/title_activity_event"
        android:parentActivityName=".Home.ImpulseActivity" >
        <meta-data
            android:name="android.support.PARENT_ACTIVITY"
            android:value=".Home.ImpulseActivity" />
    </activity>

共有1个答案

俞子实
2023-03-14

虽然问题中提到了片段,但问题实际上是向上导航。

向上导航的默认实现与标准活动的预期不完全一样。特别是,当父活动具有launchMode=“standard”(默认值)时,按向上按钮将创建它的新实例,而不是返回到上一个实例。

解决此问题有两种选择:

>

if (id == android.R.id.home)
{
    Intent intent = NavUtils.getParentActivityIntent(this);
    intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
    NavUtils.navigateUpTo(this, intent);
    return true;
}

这两种方法中的任何一种都会将旧活动带到堆栈的顶部。

 类似资料:
  • 我有亲子活动。父活动将自定义状态存储在onSaveInstanceState中(super.onSaveInstanceState是在所有需要的项目打包到bundle中之后调用的),然后在onCread中读取状态,但是出于某种原因,SavedInstanceState总是为空。 以下是父活动的日志事件序列:onPause- 我对这个问题有点不清楚——请告诉我我可能错过了什么。 谢谢

  • 我有两个活动(A1,A2)。A1呼叫A2,从A2我呼叫摄像机意图如下 它打开相机,我就可以拍照了。但是问题出现了,一旦我单击save按钮(s3中的勾选按钮),我的onActivityResult就不会被调用,而是A2的onDestroy方法被调用。我在onActivityResult fn中有一些逻辑需要完成。 我在Stackoverflow上读过一些关于这个的帖子,但我无法从中得到有用的输出。我

  • 我是Android开发新手,有一些关于生命周期活动的东西我不理解,特别是关于我正在研究的以下应用程序示例。 在我的应用程序中,我有一个登录活动和主活动。 > < li> 在我的登录活动中,如果尝试成功,将启动主活动,并完成()登录活动。< br>== 在我的主要活动中,我有一个断开连接按钮,该按钮创建了一个启动(新?)登录活动的 Intent。 == start activity(loginAct

  • 问题内容: 你们有以下问题的解决方法吗? 在注册活动的onDestroy中(当用户按下后退按钮时),我调用一个新活动,以便用户可以输入当天的某些最终生产数据,然后通过电子邮件发送该报告。问题在于,在刚刚开始的活动中,对getIntent的调用返回null,而我必须从那里获取数据。 ================================================== ========

  • MainActivity在我的应用程序中打开。考虑这个场景, MainActivity(%1)当前正在运行时。通知来了,用户点击并打开Mainactivity(2)。它起作用了。当我点击后退按钮时。Mainactivity(2)被销毁。MainActivity(1)显示给用户,应用程序冻结。我正在寻找关闭MainActivity(1)的方法。

  • 我尝试使用在viewpager中的另一个片段中添加一个片段。我收到以下错误, Java语言lang.IllegalStateException:android上的活动已被破坏。支持v4.app。碎片管理。android上的enqueueAction(FragmentManager.java:1549)。支持v4.app。回溯记录。android上的Committenternal(backbackr