碎片MyFragment无法附加到Activity?

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (10)
  • 关注 (0)
  • 查看 (221)

创建了一个小测试应用程序,它代表了我的问题。使用ActionBarSherlock来用(Sherlock)片段实现选项卡。

我的代码: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: Fragment MyFragment{410f6060} not attached to Activity

我认为这是因为同时创建了一个新的MyFragment,并且在AsyncTask完成之前附加到ActivityonPostExecute中的代码调用一个未连接的MyFragment

但是我怎样才能解决这个问题?

提问于
用户回答回答于

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

true如果该片段当前被添加到其活动中。

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

避免onPostExecute被调用时,Fragment没有附加到Activity是取消AsyncTask当暂停或停止Fragment.然后isAdded()不再有必要了。

用户回答回答于

在我的例子中,片段方法在

getActivity().onBackPressed();
用户回答回答于

如果你扩展Application初始化并维护一个静态的“global”上下文对象,如下所示,然后你可以使用它而不是该活动来加载字符串资源。

public class MyApplication extends Application {
    public static Context GLOBAL_APP_CONTEXT;

    @Override
    public void onCreate() {
        super.onCreate();
        GLOBAL_APP_CONTEXT = this;
    }
}

如果你用这个,你就能逃脱Toast和资源加载而不用担心生命周期。

用户回答回答于
if (getActivity() == null) return;

在某些情况下也有效。只需中断代码执行,并确保应用程序不崩溃。

用户回答回答于

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

调试时,当显示内容旋转时,它看起来就像PreferencesFrament的onCreate()方法被调用了两次。已经够奇怪的了。然后,我在块之外添加了isAdded()检查,它将指示崩溃,并解决了这个问题。

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

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");
                    }
                }
            }
        };
        // ...
    }
用户回答回答于

我也面临着同样的问题,我只是添加了singletone实例来获取Erick所引用的资源。

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

你也可以使用

getActivity().getResources().getString(R.string.app_name);
用户回答回答于

它们是解决这一问题和从活动中泄露碎片的相当诡计的解决方案。

因此,对于getResource或任何依赖于从片段访问活动上下文的事件,总是检查活动状态和片段状态,如下所示

 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


        }
    }
用户回答回答于

问题在于,你正在尝试使用getResources()访问资源(在本例中是字符串)。getString(),它将尝试从活动中获取资源。参见片段类的源代码:

 /**
  * 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是保存活动的对象。

由于活动可能不附加,所以getResources()调用将引发异常。

接受的解决方案IMHO并不是要走的路,因为你只是隐藏问题。正确的方法就是从其他地方获取资源,这些资源总是被保证存在,比如应用程序上下文:

youApplicationObject.getResources().getString(...)
用户回答回答于

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

Thread.sleep(2000) 

AsyncTask仍在工作,这是因为在片段重新构建之前(当你旋转时)和这个相同的AsyncTask实例(在旋转后)运行onPostExecute()之前,你没有正确地在onPostExecute()中取消AsyncTask实例,这是因为你试图用旧的片段实例(无效实例)使用getResources()查找资源:

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

相当于:

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

因此,最后的解决方案是在你旋转屏幕时,在片段重新构建之前管理AsyncTask实例(如果仍然有效,则取消),如果在转换期间取消,则借助布尔标志重新构建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;
        }

    }
}
用户回答回答于

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

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();
}

虽然我也使用了一种更复杂的方法,包括从活动中启动任务,而不是片段,但我对此并没有发现任何问题。

所属标签

可能回答问题的人

  • 西风

    renzha.net · 站长 (已认证)

    7 粉丝1 提问10 回答
  • 四无君

    0 粉丝0 提问3 回答
  • 旺仔小小鹿

    社区 · 运营 (已认证)

    46 粉丝0 提问2 回答
  • 拉布拉多拉不多

    1 粉丝0 提问2 回答

扫码关注云+社区

领取腾讯云代金券