首页 文章

片段MyFragment未附加到Activity

提问于
浏览
357

我创建了一个代表我的问题的小测试应用程序 . 我正在使用ActionBarSherlock来实现带有(Sherlock)Fragments的选项卡 .

我的代码: TestActivity.java

public class TestActivity extends SherlockFragmentActivity {
    private ActionBar actionBar;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setupTabs(savedInstanceState);
    }

    private void setupTabs(Bundle savedInstanceState) {
        actionBar = getSupportActionBar();
        actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);

        addTab1();
        addTab2();
    }

    private void addTab1() {
        Tab tab1 = actionBar.newTab();
        tab1.setTag("1");
        String tabText = "1";
        tab1.setText(tabText);
        tab1.setTabListener(new TabListener<MyFragment>(TestActivity.this, "1", MyFragment.class));

        actionBar.addTab(tab1);
    }

    private void addTab2() {
        Tab tab1 = actionBar.newTab();
        tab1.setTag("2");
        String tabText = "2";
        tab1.setText(tabText);
        tab1.setTabListener(new TabListener<MyFragment>(TestActivity.this, "2", MyFragment.class));

        actionBar.addTab(tab1);
    }
}

TabListener.java

public class TabListener<T extends SherlockFragment> implements com.actionbarsherlock.app.ActionBar.TabListener {
    private final SherlockFragmentActivity mActivity;
    private final String mTag;
    private final Class<T> mClass;

    public TabListener(SherlockFragmentActivity activity, String tag, Class<T> clz) {
        mActivity = activity;
        mTag = tag;
        mClass = clz;
    }

    /* The following are each of the ActionBar.TabListener callbacks */

    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        SherlockFragment preInitializedFragment = (SherlockFragment) mActivity.getSupportFragmentManager().findFragmentByTag(mTag);

        // Check if the fragment is already initialized
        if (preInitializedFragment == null) {
            // If not, instantiate and add it to the activity
            SherlockFragment mFragment = (SherlockFragment) SherlockFragment.instantiate(mActivity, mClass.getName());
            ft.add(android.R.id.content, mFragment, mTag);
        } else {
            ft.attach(preInitializedFragment);
        }
    }

    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        SherlockFragment preInitializedFragment = (SherlockFragment) mActivity.getSupportFragmentManager().findFragmentByTag(mTag);

        if (preInitializedFragment != null) {
            // Detach the fragment, because another one is being attached
            ft.detach(preInitializedFragment);
        }
    }

    public void onTabReselected(Tab tab, FragmentTransaction ft) {
        // User selected the already selected tab. Usually do nothing.
    }
}

MyFragment.java

public class MyFragment extends SherlockFragment {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        new AsyncTask<Void, Void, Void>() {

            @Override
            protected Void doInBackground(Void... params) {
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException ex) {
                }
                return null;
            }

            @Override
            protected void onPostExecute(Void result){
                getResources().getString(R.string.app_name);
            }

        }.execute();
    }
}

我添加了 Thread.sleep 部分来模拟下载数据 . onPostExecute 中的代码是模拟 Fragment 的使用 .

当我在横向和纵向之间快速旋转屏幕时,我在 onPostExecute 代码处得到一个异常:

java.lang.IllegalStateException:片段MyFragment {410f6060}未附加到Activity

我认为这是因为在此期间创建了一个新的 MyFragment ,并且在 AsyncTask 结束之前附加到了Activity . onPostExecute 中的代码调用未附加的 MyFragment .

但是我该如何解决这个问题呢?

11 回答

  • 21

    一个老帖子,但我对最高投票的答案感到惊讶 .

    对此的正确解决方案应该是取消onStop中的asynctask(或在片段中的任何适当位置) . 这样您就不会引入内存泄漏(asynctask保留对已销毁片段的引用),并且您可以更好地控制片段中发生的事情 .

    @Override
    public void onStop() {
        super.onStop();
        mYourAsyncTask.cancel(true);
    }
    
  • 10
    if (getActivity() == null) return;
    

    在某些情况下也有效 . 只需打破代码执行,确保应用程序不会崩溃

  • 0

    问题是您正在尝试使用getResources() . getString()来访问资源(在本例中为字符串),它将尝试从Activity获取资源 . 查看Fragment类的源代码:

    /**
      * Return <code>getActivity().getResources()</code>.
      */
     final public Resources getResources() {
         if (mHost == null) {
             throw new IllegalStateException("Fragment " + this + " not attached to Activity");
         }
         return mHost.getContext().getResources();
     }
    

    mHost 是保存您的Activity的对象 .

    因为可能没有附加Activity,所以getResources()调用将抛出异常 .

    可接受的解决方案恕我直言,因为你只是隐藏问题 . 正确的方法是从其他地方获取始终保证存在的资源,例如应用程序上下文:

    youApplicationObject.getResources().getString(...)
    
  • 14

    如果扩展 Application 类并维护静态'global' Context对象(如下所示),则可以使用该对象而不是活动来加载String资源 .

    public class MyApplication extends Application {
        public static Context GLOBAL_APP_CONTEXT;
    
        @Override
        public void onCreate() {
            super.onCreate();
            GLOBAL_APP_CONTEXT = this;
        }
    }
    

    如果你使用它,你可以逃避 Toast 和资源加载而不用担心生命周期 .

  • 24

    在我的例子中,片段方法已被调用

    getActivity().onBackPressed();
    
  • 18

    我在这里遇到了两种不同的情况:

    1)当我希望异步任务完成时:假设我的onPostExecute确实存储了收到的数据,然后调用一个监听器来更新视图,为了更高效,我希望任务完成任务,这样我就可以在用户出现时准备好数据了背部 . 在这种情况下,我通常这样做:

    @Override
    protected void onPostExecute(void result) {
        // do whatever you do to save data
        if (this.getView() != null) {
            // update views
        }
    }
    

    2)当我只想在可以更新视图时完成异步任务时:你在这里建议的情况下,任务只更新视图,不需要数据存储,所以如果视图是完成的,它就没有完成任务的线索不再被展示 . 我这样做:

    @Override
    protected void onStop() {
        // notice here that I keep a reference to the task being executed as a class member:
        if (this.myTask != null && this.myTask.getStatus() == Status.RUNNING) this.myTask.cancel(true);
        super.onStop();
    }
    

    我发现没有问题,虽然我也使用(可能)更复杂的方式,包括从活动而不是片段启动任务 .

    希望这有助于某人! :)

  • 0

    我遇到了同样的问题,我只需添加单一实例来获取Erick所引用的资源

    MainFragmentActivity.defaultInstance().getResources().getString(R.string.app_name);
    

    你也可以用

    getActivity().getResources().getString(R.string.app_name);
    

    我希望这将有所帮助 .

  • 2

    代码的问题在于您使用AsyncTask的方式,因为在睡眠线程期间旋转屏幕时:

    Thread.sleep(2000)
    

    AsyncTask仍在工作,这是因为你没有在片段重建(旋转时)之前在onDestroy()中正确取消AsyncTask实例,并且当这个相同的AsyncTask实例(在旋转之后)运行onPostExecute()时,这会尝试查找getResources()与旧片段实例(无效实例)的资源:

    getResources().getString(R.string.app_name)
    

    这相当于:

    MyFragment.this.getResources().getString(R.string.app_name)
    

    所以最终的解决方案是在旋转屏幕时重建片段之前管理AsyncTask实例(如果这仍然有效则取消),如果在转换期间取消,则在重建后借助布尔标志重新启动AsyncTask:

    public class MyFragment extends SherlockFragment {
    
        private MyAsyncTask myAsyncTask = null;
        private boolean myAsyncTaskIsRunning = true;
    
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            if(savedInstanceState!=null) {
                myAsyncTaskIsRunning = savedInstanceState.getBoolean("myAsyncTaskIsRunning");
            }
            if(myAsyncTaskIsRunning) {
                myAsyncTask = new MyAsyncTask();
                myAsyncTask.execute();
            }
        }
    
        @Override
        public void onSaveInstanceState(Bundle outState) {
            super.onSaveInstanceState(outState);
            outState.putBoolean("myAsyncTaskIsRunning",myAsyncTaskIsRunning);
        }
    
        @Override
        public void onDestroy() {
            super.onDestroy();
            if(myAsyncTask!=null) myAsyncTask.cancel(true);
            myAsyncTask = null;
    
        }
    
        public class MyAsyncTask extends AsyncTask<Void, Void, Void>() {
    
            public MyAsyncTask(){}
    
            @Override
            protected void onPreExecute() {
                super.onPreExecute();
                myAsyncTaskIsRunning = true;
            }
            @Override
            protected Void doInBackground(Void... params) {
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException ex) {}
                return null;
            }
    
            @Override
            protected void onPostExecute(Void result){
                getResources().getString(R.string.app_name);
                myAsyncTaskIsRunning = false;
                myAsyncTask = null;
            }
    
        }
    }
    
  • 7

    Their are quite trick solution for this and leak of fragment from activity.

    因此,对于getResource或依赖于从Fragment访问的活动上下文的任何东西,它总是检查活动状态和片段状态如下

    Activity activity = getActivity(); 
        if(activity != null && isAdded())
    
             getResources().getString(R.string.no_internet_error_msg);
    //Or any other depends on activity context to be live like dailog
    
    
            }
        }
    
  • 730

    我找到了一个非常简单的答案:isAdded()

    如果片段当前已添加到其活动,则返回true .

    @Override
    protected void onPostExecute(Void result){
        if(isAdded()){
            getResources().getString(R.string.app_name);
        }
    }
    

    Fragment 未附加到 Activity 时,为了避免 onPostExecute 被取消,是在暂停或停止 Fragment 时取消 AsyncTask . 那么 isAdded() 就不再需要了 . 但是,建议将此检查保留在原位 .

  • 0

    当带有加载的首选项的应用程序设置活动可见时,我遇到了类似的问题 . 如果我要更改其中一个首选项然后再旋转显示内容并再次更改首选项,它会崩溃,并显示片段(我的Preferences类)未附加到活动 .

    调试时,看起来像在旋转显示内容时两次调用PreferencesFragment的onCreate()方法 . 这已经很奇怪了 . 然后我在块外面添加了isAdded()检查,它会指示崩溃并解决了问题 .

    以下是更新首选项摘要以显示新条目的侦听器代码 . 它位于我的Preferences类的onCreate()方法中,它扩展了PreferenceFragment类:

    public static class Preferences extends PreferenceFragment {
        SharedPreferences.OnSharedPreferenceChangeListener listener;
    
        @Override
        public void onCreate(Bundle savedInstanceState) {
            // ...
            listener = new SharedPreferences.OnSharedPreferenceChangeListener() {
                @Override
                public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
                    // check if the fragment has been added to the activity yet (necessary to avoid crashes)
                    if (isAdded()) {
                        // for the preferences of type "list" set the summary to be the entry of the selected item
                        if (key.equals(getString(R.string.pref_fileviewer_textsize))) {
                            ListPreference listPref = (ListPreference) findPreference(key);
                            listPref.setSummary("Display file content with a text size of " + listPref.getEntry());
                        } else if (key.equals(getString(R.string.pref_fileviewer_segmentsize))) {
                            ListPreference listPref = (ListPreference) findPreference(key);
                            listPref.setSummary("Show " + listPref.getEntry() + " bytes of a file at once");
                        }
                    }
                }
            };
            // ...
        }
    

    我希望这会对别人有所帮助!

相关问题