当快速切换ActionBar选项卡时,没有找到片段id的视图



我正在使用兼容性库为平板电脑和而不是开发Android应用程序。

只有一个活动,它使用3个选项卡的动作栏。在TabListener中,我使用setContentView来加载特定于该选项卡的布局,然后将相关片段添加到它们的框架中。这个几乎完全像我想要的那样工作,除了当你在标签之间切换得足够快时,应用程序会崩溃。

我使用三星Galaxy Tab作为我的调试设备,切换标签非常快。在正常的速度下,我可以在它们之间来回点击,页面就会立即加载。问题是当我在标签之间过度切换时。

起初我得到一个

IllegalStateException: Fragment not added

如下所示:http://code.google.com/p/android/issues/detail?id=17029按照在onTabUnselected中使用try/catch块的建议,我使应用程序更加健壮,但这导致了手头的问题:

IllegalArgumentException: No view found for id 0x... for fragment ...

我没有在网上找到任何其他有同样问题的人的案例,所以我担心我可能正在做一些不支持的事情。再次,我要做的是在一个活动中使用3种不同的布局-当你点击一个选项卡时,监听器会调用setContentView来改变布局,然后添加片段。它工作得很好,除非你开始频繁地在选项卡之间切换。

我得到了这个想法:http://developer.android.com/resources/samples/ApiDemos/src/com/example/android/apis/app/FragmentTabs.html而不是TabListener保持一个片段的引用,我有一个数组。此外,我没有使用附加/分离,因为这些只是在API 13中添加的。

我的理论是setContentView没有完成创建视图,这就是为什么FragmentTransaction不能添加它们,或者当另一个选项卡被选中和setContentView被调用时,正在为一个选项卡添加片段,破坏另一组视图。

我试图破解一些东西来减慢标签切换,但没有得到任何地方。

下面是TabListener的代码:
private class BTabListener<T extends Fragment> implements ActionBar.TabListener{
    private int mLayout;
    private Fragment[] mFrags;
    private TabData mTabData;
    private Activity mAct;
    private boolean mNoNewFrags;

    public BTabListener(Activity act, int layout, TabData td, boolean frags){
        mLayout = layout;
        mTabData = td;
        mAct = act;
        mNoNewFrags = frags;
        mFrags = new Fragment[mTabData.fragTags.length];
        for(int i=0; i<mFrags.length; i++){
            //on an orientation change, this will find the fragments that were recreated by the system
            mFrags[i] = mAct.getFragmentManager().findFragmentByTag(mTabData.fragTags[i]);
        }
    }
    @Override
    public void onTabReselected(Tab tab, FragmentTransaction ft) {
    }
    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        //this gets called _after_ unselected
        //note: unselected wont have been called after an orientation change!
        //we also need to watch out because tab 0 always gets selected when adding the tabs
        //set the view for this tab
        mAct.setContentView(mLayout);
        for(int i=0; i<mFrags.length; i++){
            //this will be null when the tab is first selected
            if(mFrags[i]==null ){
                mFrags[i] = Fragment.instantiate(GUITablet.this, mTabData.classes[i].getName());                    
            }
            //if there was an orientation change when we were on this page, the fragment is already added
            if(!mNoNewFrags || mDefaultTab!=tab.getPosition()){
                ft.add(mTabData.containterIDs[i], mFrags[i], mTabData.fragTags[i]);
            }
        }
        mNoNewFrags = false;

    }
    @Override
    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        // this gets called when another tab is selected, before it's onSelected method 
        for(Fragment f : mFrags){
            try{ //extra safety measure
                ft.remove(f);
            }catch(Exception e){
                e.printStackTrace();
                System.out.println("unselect couldnt remove");
            }
        }
    }
}

最后,堆栈跟踪:

09-29 01:53:08.200: ERROR/AndroidRuntime(4611): java.lang.IllegalArgumentException: No view found for id 0x7f0b0078 for fragment Fraggle{40ab2230 #2 id=0x7f0b0078 dummy2}
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:729)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:926)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.BackStackRecord.run(BackStackRecord.java:578)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1226)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.FragmentManagerImpl$1.run(FragmentManager.java:374)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.os.Handler.handleCallback(Handler.java:587)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.os.Handler.dispatchMessage(Handler.java:92)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.os.Looper.loop(Looper.java:132)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at android.app.ActivityThread.main(ActivityThread.java:4028)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at java.lang.reflect.Method.invokeNative(Native Method)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at java.lang.reflect.Method.invoke(Method.java:491)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:844)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611):     at dalvik.system.NativeStart.main(Native Method)

谢谢! !

好的,我找到了一个方法:

把对片段的引用放在布局文件中,并在一个try/catch块中包围onTabSelected中的setContentView调用。

异常处理已经处理好了!

我知道这是一个有点老的问题,但我有相同的,并找到了不同的工作。

方法本身描述在这里

避免在执行选项卡切换时重新创建相同的视图

但是在这个特定的崩溃的上下文中,做Show/Hide而不是add/replace可以避免对片段的onCreateView进行多次快速调用。

最后的代码是这样的:

@Override
public void onTabSelected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
if (fragments[tab.getPosition()] == null) {
     switch(tag.getPosition()){
        case 0: fragments[tab.getPosition()] = new // fragment for this position
        break;
        // repeat for all the tabs, for each `case`
     }
     fragmentTransaction.add(R.id.container, fragments[tab.getPosition()]);
}else{
     fragmentTransaction.show(fragments[tab.getPosition()]);
}
 }
@Override
public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
if (fragments[tab.getPosition()] != null)
    fragmentTransaction.hide(fragments[tab.getPosition()]);
}

在Google Play崩溃报告中看到类似的情况。

java.lang.IllegalStateException:
    Fragment not added: MessageDisplayFragment{406e28f0 #2 id=0x7f0b0020}
at android.app.BackStackRecord.hide(BackStackRecord.java:397)
at org.kman.AquaMail.ui.AccountListActivity$UIMediator_API11_TwoPane.
    closeMessageDisplay(AccountListActivity.java:2585)
相关代码:

AbsMessageFragment fragDisplay = (AbsMessageFragment)
    mFM.findFragmentById(R.id.fragment_id_message_display);
if (fragDisplay != null) {
    FragmentTransaction ft = mFM.beginTransaction();
    ft.setTransition(FragmentTransaction.TRANSIT_NONE);
    ft.show(some other fragment);
    ft.hide(fragDisplay);

^^这就是它崩溃的地方

片段肯定在那里(它是由findFragmentById返回的),但调用hide()爆炸了"IllegalStateException: fragment not added"

不添加呢?如果findFragmentById能够找到它,它是如何不添加的?

所有的FragmentManager调用都是从UI线程进行的。

被删除的片段(fragDisplay)是早些时候添加的,我确信它的FragmentTransaction是提交的。

通过循环遍历我的片段,删除任何已添加的片段,然后添加新的片段来修复它。还要先检查一下,你没有试图用null或现有的片段来替换它。

编辑:看起来只是

getChildFragmentManager().executePendingTransactions();

阻止它崩溃

private void replaceCurrentTabFragment(TabFragment tabFragment){
    if (tabFragment == null){ return; }
    if (tabFragment == getActiveTabFragment()){ return; }
    FragmentTransaction ft = getChildFragmentManager().beginTransaction();
    for (TabFragment fragment : mTabButtons.values()){
        if (((Fragment)fragment).isAdded()){
            ft.remove((Fragment)fragment);
        }
    }
    ft.add(R.id.fragment_frameLayout, (Fragment) tabFragment);
    ft.commit();
    getChildFragmentManager().executePendingTransactions();
}
private TabFragment getActiveTabFragment(){
    return (TabFragment) getChildFragmentManager().findFragmentById(R.id.fragment_frameLayout);
}

最新更新