片段 MyFragment 未附加到 Activity

Fragment MyFragment not attached to Activity

提问人:nhaarman 提问时间:6/7/2012 更新时间:8/27/2022 访问量:311944

问:

我创建了一个小型测试应用程序来代表我的问题。 我正在使用 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.sleeponPostExecuteFragment

当我在横向和纵向之间非常快速地旋转屏幕时,我在代码中出现异常:onPostExecute

java.lang.IllegalStateException:Fragment MyFragment{410f6060} 不是 附加到活动

我认为这是因为在此期间创建了一个新的,并且在完成之前附加到活动。中的代码调用未附加的 .MyFragmentAsyncTaskonPostExecuteMyFragment

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

android-fragments actionbarsSherlock

评论

1赞 foxis 3/16/2017
您应该使用从片段充气的视图。 现在,当您想要获取资源时,请使用此视图:.它帮助我修复这个错误。mView = inflater.inflate(R.layout.my_layout, container, false) mView.getResources().***
0赞 nhaarman 3/16/2017
@foxis 这会泄漏附加到您的“mView”的内容。Context
0赞 foxis 3/17/2017
可能是我还没有检查。为避免泄漏,如何在onDestroy中获取null?mView

答:

842赞 nhaarman 6/8/2012 #1

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

如果片段当前已添加到其 Activity 中,则返回。true

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

为了避免在 未附加到 时被调用,请在暂停或停止 时取消 。那么就没有必要了。但是,建议保持此检查到位。onPostExecuteFragmentActivityAsyncTaskFragmentisAdded()

评论

0赞 CoDe 10/28/2013
就我而言,当我从以下位置启动另一个应用程序 Intent...然后我遇到同样的错误......有什么建议吗?
2赞 Lucas Jota 3/21/2014
developer.android.com/reference/android/app/.........还有,它是在 API 级别 13 上添加的isDetached()
5赞 nhaarman 3/22/2014
在 API <11 上时,您正在使用它工作的地方 developer.android.com/reference/android/support/v4/app/...
0赞 Ataru 11/17/2014
当我使用DialogFragment时,我遇到了这个问题。关闭dialogFragment后,我尝试启动另一个活动。然后发生了这个错误。我通过在startActivity之后调用dismiss()来避免此错误。问题是片段已经从 Activity 中分离出来。
26赞 luixal 4/24/2013 #2

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

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

我发现这没有问题,尽管我也使用了一种(可能)更复杂的方式,包括从活动而不是片段启动任务。

希望这对某人有所帮助!:)

18赞 Erick Reátegui Diaz 5/30/2014 #3

代码的问题在于使用 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;
        }

    }
}

评论

0赞 Prabs 2/6/2017
相反,如果使用帮助getResources().***Fragments.this.getResource().***
10赞 Aristo Michael 8/27/2014 #4

我遇到了同样的问题,我只是添加单调实例来获取埃里克提到的资源

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

您还可以使用

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

我希望这会有所帮助。

1赞 Anthony Chuinard 6/9/2015 #5

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

public class MyApplication extends Application {
    public static Context GLOBAL_APP_CONTEXT;

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

如果你使用它,你可以摆脱资源加载,而不必担心生命周期。Toast

评论

7赞 Anthony Chuinard 6/15/2015
我被否决了,但没有人解释原因。静态上下文通常很糟糕,但我认为,如果您有静态应用程序引用,这不是内存泄漏。
1赞 Vivek Kumar Srivastava 12/26/2017
您的答案被否决了,因为这只是一个黑客,而不是正确的解决方案。查看@nhaarman共享的解决方案
2赞 Marcell 7/11/2015 #6

当加载首选项的应用程序设置活动可见时,我遇到了类似的问题。如果我更改其中一个首选项,然后使显示内容旋转并再次更改首选项,它将崩溃并显示一条消息,指出片段(我的 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");
                    }
                }
            }
        };
        // ...
    }

我希望这对其他人有所帮助!

38赞 Bitcoin Cash - ADA enthusiast 7/1/2016 #7

问题在于您正在尝试使用 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() 调用将引发 Exception。

恕我直言,公认的解决方案不是要走的路,因为您只是隐藏了问题。正确的方法是从始终保证存在的其他位置获取资源,例如应用程序上下文:

youApplicationObject.getResources().getString(...)

评论

0赞 Geekarist 5/29/2017
我之所以使用这个解决方案,是因为当我的片段暂停时,我需要执行。谢谢getString()
0赞 MGLabs 4/9/2022
更新:来自文档,“在 Build.VERSION_CODES#R 之后,必须通过活动或上下文获取资源。Application#getResources() 可能会在多窗口或辅助显示器上报告错误值。您可以改为使用 Context.getResources() 获取 Resources 的实例。
24赞 Vinayak 9/2/2016 #8

他们是一个非常棘手的解决方案,并且从活动中泄漏碎片。

因此,如果 getResource 或任何依赖于从 Fragment 访问的活动上下文,它始终检查活动状态和 Fragments 状态,如下所示

 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


        }
    }

评论

7赞 NguyenDat 11/30/2016
isAdded() 就足够了,因为: final public boolean isAdded() { return mHost != null && mAdded; }
0赞 David 12/9/2019
就我而言,此检查不是 enoug,尽管我添加了这些检查,但仍然崩溃。
0赞 CoolMind 4/20/2020
@David,就够了。我从未见过崩溃的情况。是否确定已显示活动并附加了片段?isAddedgetString()isAdded == true
0赞 CoolMind 10/18/2016 #9

在我的情况下,片段方法已被调用

getActivity().onBackPressed();
0赞 Raz 1/5/2017 #10

一篇旧帖子,但我对投票最多的答案感到惊讶。

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

@Override
public void onStop() {
    super.onStop();
    mYourAsyncTask.cancel(true);
}

评论

1赞 nhaarman 1/5/2017
最受好评的答案包括这一点。此外,可能无法阻止被调用。cancelonPostExecute
0赞 Raz 1/21/2017
调用 cancel 可以保证永远不会调用 onPostExecute,这两个调用都在同一线程上执行,因此可以保证在调用 cancel 后不会调用它
15赞 superUser 7/5/2017 #11
if (getActivity() == null) return;

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

-1赞 Fakhriddin Abdullaev 10/25/2020 #12

简单的解决方案和工作 100%

if (getActivity() == null || !isAdded()) return;
1赞 Dilanka Fernando 1/28/2021 #13

我在 Xamarine Android 中遇到了类似的错误消息“Fragment MyFragment not attached to Context”。

由于此资源调用而出现此错误消息

button.Text = Resources.GetString(Resource.String.please_wait)

我确实通过在Xamarine Android中使用来解决这个问题。

if (Context != null && IsAdded){ 
    button.Text = Resources.GetString(Resource.String.please_wait);
}
0赞 Jose 8/27/2022 #14

将此添加到您的Fragemnt

Activity activity;
@Override
public void onAttach(@NonNull Context context) {
    super.onAttach(context);
    activity = context instanceof Activity ? (Activity) context : null;
}

然后将 getContext() , getActivity() , requireActivity() 或 requireContext() 更改为 activity