片段 MyFragment 未附加到 Activity
Posted
技术标签:
【中文标题】片段 MyFragment 未附加到 Activity【英文标题】:Fragment MyFragment not attached to Activity 【发布时间】:2012-06-10 18:22:12 【问题描述】:我创建了一个代表我的问题的小型测试应用程序。 我正在使用 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: 片段 MyFragment410f6060 不是 附加到活动
我认为这是因为同时创建了一个新的MyFragment
,并且在AsyncTask
完成之前附加到了Activity。 onPostExecute
中的代码调用未附加的 MyFragment
。
但是我该如何解决这个问题呢?
【问题讨论】:
你应该使用片段充气器的视图。mView = inflater.inflate(R.layout.my_layout, container, false)
现在当你想获取资源时使用这个视图:mView.getResources().***
。它帮助我修复了这个错误。
@foxis 泄漏了附加到您的 mView 的 Context
。
可能我还没有检查。为避免泄漏如何在 onDestroy 中获取 null mView
?
【参考方案1】:
我找到了非常简单的答案:isAdded()
:
如果片段当前已添加到其活动中,则返回
true
。
@Override
protected void onPostExecute(Void result)
if(isAdded())
getResources().getString(R.string.app_name);
为了避免在Fragment
未附加到Activity
时调用onPostExecute
,请在暂停或停止Fragment
时取消AsyncTask
。那么isAdded()
就不再需要了。但是,建议保留此检查。
【讨论】:
在我的情况下,当我从...启动另一个应用程序 Intent 时,我得到了同样的错误...任何建议? developer.android.com/reference/android/app/… ...还有isDetached()
,它是在 API 级别 13 上添加的
在 API developer.android.com/reference/android/support/v4/app/…,它可以工作。
我在使用 DialogFragment 时遇到了这个问题。关闭 dialogFragment 后,我尝试开始另一个活动。然后出现了这个错误。我通过在 startActivity 之后调用dismiss() 避免了这个错误。问题是该片段已经与 Activity 分离。【参考方案2】:
问题是您正在尝试使用 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(...)
【讨论】:
我使用了这个解决方案,因为我需要在我的片段暂停时执行getString()
。谢谢【参考方案3】:
我在这里遇到了两种不同的情况:
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();
我发现这没有问题,尽管我也使用了(可能)更复杂的方式,包括从活动而不是片段启动任务。
希望这对某人有所帮助! :)
【讨论】:
【参考方案4】:他们是相当巧妙的解决方案和活动中的片段泄漏。
因此,在 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
【讨论】:
isAdded() 就足够了,因为: final public boolean isAdded() return mHost != null && mAdded; 在我的情况下,这个检查还不够格,尽管我添加了这些检查仍然会崩溃。 @David,isAdded
就够了。如果isAdded == true
发生getString()
崩溃,我从未见过这种情况。您确定显示了活动并附加了片段吗?【参考方案5】:
您的代码的问题在于您使用 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;
【讨论】:
如果getResources().***
使用Fragments.this.getResource().***
有帮助【参考方案6】:
if (getActivity() == null) return;
在某些情况下也有效。只需中断代码执行并确保应用程序不会崩溃
【讨论】:
【参考方案7】:我遇到了同样的问题,我只是添加了单例实例来获取 Erick 引用的资源
MainFragmentActivity.defaultInstance().getResources().getString(R.string.app_name);
你也可以使用
getActivity().getResources().getString(R.string.app_name);
我希望这会有所帮助。
【讨论】:
【参考方案8】:当带有已加载首选项的应用程序设置活动可见时,我遇到了类似的问题。如果我要更改其中一个首选项,然后让显示内容旋转并再次更改首选项,它将崩溃并显示片段(我的 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");
;
// ...
我希望这对其他人有帮助!
【讨论】:
【参考方案9】:如果您扩展Application
类并维护一个静态“全局”上下文对象,如下所示,那么您可以使用它而不是活动来加载字符串资源。
public class MyApplication extends Application
public static Context GLOBAL_APP_CONTEXT;
@Override
public void onCreate()
super.onCreate();
GLOBAL_APP_CONTEXT = this;
如果你使用它,你可以摆脱Toast
和资源加载,而不用担心生命周期。
【讨论】:
我被否决了,但没有人解释原因。静态上下文通常很糟糕,但我认为如果您有静态应用程序引用,这不是内存泄漏。 您的答案被否决了,因为这只是一个黑客而不是正确的解决方案。检查@nhaarman 分享的解决方案【参考方案10】:在我的例子中,片段方法是在之后调用的
getActivity().onBackPressed();
【讨论】:
【参考方案11】:一个旧帖子,但我对投票最多的答案感到惊讶。
对此的正确解决方案应该是取消 onStop 中的异步任务(或在您的片段中的任何适当位置)。这样您就不会引入内存泄漏(保持对已销毁片段的引用的异步任务),并且您可以更好地控制片段中发生的事情。
@Override
public void onStop()
super.onStop();
mYourAsyncTask.cancel(true);
【讨论】:
最受好评的答案包括这个。此外,cancel
可能不会阻止 onPostExecute
被调用。
调用cancel确实保证永远不会调用onPostExecute,两个调用都在同一个线程上执行,因此你保证在调用cancel后它不会被调用【参考方案12】:
我在 Xamarine Android 中收到了类似的错误消息 “Fragment MyFragment not attach to Context”。
由于此资源调用而出现此错误消息
button.Text = Resources.GetString(Resource.String.please_wait)
我确实通过在 Xamarine Android 中使用解决了这个问题。
if (Context != null && IsAdded)
button.Text = Resources.GetString(Resource.String.please_wait);
【讨论】:
【参考方案13】:简单的解决方案,百分百有效
if (getActivity() == null || !isAdded()) return;
【讨论】:
以上是关于片段 MyFragment 未附加到 Activity的主要内容,如果未能解决你的问题,请参考以下文章