问答中心分类: ANDROID片段 MyFragment 未附加到 Activity
0
匿名用户 提问 14分钟 前

我创建了一个代表我的问题的小型测试应用程序。我正在使用 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(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(TestActivity.this, "2", MyFragment.class));

        actionBar.addTab(tab1);
    }
}

TabListener.java

public class TabListener implements com.actionbarsherlock.app.ActionBar.TabListener {
    private final SherlockFragmentActivity mActivity;
    private final String mTag;
    private final Class mClass;

    public TabListener(SherlockFragmentActivity activity, String tag, Class 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() {

            @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在此期间已创建,并在之前附加到 ActivityAsyncTask完成的。中的代码onPostExecute呼吁一个独立的MyFragment.
但是我该如何解决这个问题?

foxis 回复 14分钟 前

您应该使用片段充气机的视图。mView = inflater.inflate(R.layout.my_layout, container, false)现在当你想获取资源时使用这个视图:mView.getResources().***.它帮助我修复了这个错误。

foxis 回复 14分钟 前

@foxis 泄露了Context附加到您的`mView`。

foxis 回复 14分钟 前

可能我还没有检查。为避免泄漏如何获取 nullmView在 onDestroy 中?

13 Answers
0
Bitcoin Cash - ADA enthusiast 回答 14分钟 前

问题是您正在尝试使用 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 可能未附加,所以您的 getResources() 调用将引发异常。
接受的解决方案恕我直言,因为您只是在隐藏问题,因此不是可行的方法。正确的方法是从其他始终保证存在的地方获取资源,例如应用程序上下文:

youApplicationObject.getResources().getString(...)
Geekarist 回复 14分钟 前

我使用了这个解决方案,因为我需要执行getString()当我的片段被暂停时。谢谢

MGLabs 回复 14分钟 前

更新:从文档中,“在 Build.VERSION_CODES#R 之后,必须通过 Activity 或 Context 获取资源。Application#getResources() 可能会在多窗口或辅助显示器上报告错误的值。”您可以改为使用 Context.getResources() 获取资源实例。

0
luixal 回答 14分钟 前

我在这里遇到了两种不同的情况:
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
Vinayak 回答 14分钟 前

他们是非常巧妙的解决方案,并且会从活动中泄漏片段。
因此,在 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


        }
    }
NguyenDat 回复 14分钟 前

isAdded() 就足够了,因为: final public boolean isAdded() { return mHost != null && mAdded; }

David 回复 14分钟 前

在我的情况下,这些检查还不够格,尽管我添加了这些,但仍然会崩溃。

CoolMind 回复 14分钟 前

@大卫,isAdded足够的。我从没见过这样的情况getString()如果坠毁了isAdded == true.您确定显示了活动并附加了片段吗?

0
Erick Reátegui Diaz 回答 14分钟 前

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

    }
}
Prabs 回复 14分钟 前

相反,如果getResources().***使用Fragments.this.getResource().***帮助

0
superUser 回答 14分钟 前
if (getActivity() == null) return;

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