我已经创建了一个代表我的问题的小测试应用程序。我正在使用ActionBarSherlock来实现带有(神探夏洛克)片段的标签。
我的代码: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
完成之前附加到了该活动。onPostExecute
中的代码调用未附加的MyFragment
。
但是我该如何解决这个问题呢?
发布于 2012-06-08 07:51:23
我找到了非常简单的答案:isAdded()
如果片段当前已添加到其活动中,则
返回
true
。
@Override
protected void onPostExecute(Void result){
if(isAdded()){
getResources().getString(R.string.app_name);
}
}
为了避免在Fragment
未连接到Activity
时调用onPostExecute
,就是在暂停或停止Fragment
时取消AsyncTask
。那么就不再需要isAdded()
了。但是,建议将此检查保留在适当的位置。
发布于 2016-07-01 05:00:56
问题是您试图使用getResources().getString()访问资源(在本例中是字符串),它将尝试从活动中获取资源。请参阅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
是保存您的活动的对象。
由于活动可能未附加,因此getResources()调用将抛出异常。
被接受的解决方案IMHO不是可行的方法,因为你只是在隐藏问题。正确的方法是从确保始终存在的其他地方获取资源,比如应用程序上下文:
youApplicationObject.getResources().getString(...)
发布于 2013-04-24 18:13:35
我在这里遇到了两种不同的情况:
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();
}
我没有发现这方面的问题,尽管我还使用了一种(可能)更复杂的方法,包括从活动而不是从片段启动任务。
希望这对某些人有帮助!:)
https://stackoverflow.com/questions/10919240
复制相似问题