IllegalArgumentException:在ViewPager中找不到用于片段--- ViewPager的id的视图

Xie*_*eyi 22 android android-fragments android-viewpager

我遇到了困扰我好几天的问题.

ViewPager主要活动中有一个持有3 Fragment秒作为标签片段.在第一个片段中有一个ListView包含一些视图,哪个是最重要的视图,另一个视图ViewPager.我想在子中保存一些照片ViewPager,并在这里使用更多片段.

现在有麻烦:
当第一个 Fragment停止时(父节点中的第三个片段在ViewPager屏幕上显示)并恢复(用户切换到第二个片段),应用程序崩溃,调试器说:

java.lang.IllegalArgumentException: No view found for id 0x7f05008b (com.example.viewpager:id/sub_viewpager) for fragment ScreenSlidePageFragment
Run Code Online (Sandbox Code Playgroud)

我已经使用了,getChildFragmentManager()因为这是嵌套片段的情况.

以下是与父ViewPager中第一个片段对应的列表适配器的关键代码:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    int type = getItemViewType(position);
    switch (type) {
        case TYPE_BANNER:
            if (convertView == null) {
                convertView = mBannerView.getBannerView(parent);
            }
            mBannerView.update(convertView);
            break;
        case TYPE_ITEM:
            break;
    }
    return convertView;
}
Run Code Online (Sandbox Code Playgroud)

这是代码mBannerView:

public class BannerView {

    private static final DisplayImageOptions IMAGE_OPTIONS_SCALE_STRETCHED =
            new DisplayImageOptions.Builder()
                    .cacheInMemory()
                    .cacheOnDisc()
                    .imageScaleType(ImageScaleType.EXACTLY_STRETCHED)
                    .build();

    private FragmentActivity mActivity;
    private Fragment mFragment;
    private List<Banner> mBanners;
    private ScreenSlidePagerAdapter mPagerAdapter;
    private ViewPager mViewPager;

    public BannerView(FragmentActivity activity, Fragment fragment) {
        mActivity = activity;
        mFragment = fragment;
    }

    public void update(View convertView) {
        mViewPager = (ViewPager) convertView;
        if (mBanners != null && !mBanners.isEmpty()) {
            if (mPagerAdapter == null) {
                mPagerAdapter = new ScreenSlidePagerAdapter(mFragment.getChildFragmentManager());
                mViewPager.setAdapter(mPagerAdapter);
            }
        }
        mViewPager.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (mOnBannerClickListener != null) {
                    mOnBannerClickListener.onBannerClick();
                }
            }
        });
    }

    class ScreenSlidePagerAdapter extends FragmentStatePagerAdapter {
        public ScreenSlidePagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return new ScreenSlidePageFragment(mBanners.get(position).getImageUrl());
        }

        @Override
        public int getCount() {
            return mBanners == null ? 0 : mBanners.size();
        }
    }

    class ScreenSlidePageFragment extends Fragment {

        private String mUrl;

        ScreenSlidePageFragment(String url) {
            super();
            mUrl = url;
        }

        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container,
                                 Bundle savedInstanceState) {
            View view = inflater.inflate(R.layout.item_banner, container, false);
            if (view != null) {
                ImageView imageView = (ImageView) view.findViewById(R.id.item_banner_image);
                imageView.setLayoutParams(new LinearLayout.LayoutParams(
                        LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT));
                ImageLoader.getInstance().displayImage(mUrl, imageView, IMAGE_OPTIONS_SCALE_STRETCHED);
            }
            return view;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

这是详细的错误列表:

11-10 18:12:19.217    1444-1444/? E/MessageQueue-JNI? java.lang.IllegalArgumentException: No view found for id 0x7f05008b (com.example.viewpager:id/sub_viewpager) for fragment ScreenSlidePageFragment{428d8ea0 #0 id=0x7f05008b}
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:919)
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1104)
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:1086)
        at android.support.v4.app.FragmentManagerImpl.dispatchActivityCreated(FragmentManager.java:1884)
        at android.support.v4.app.Fragment.performActivityCreated(Fragment.java:1514)
        at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:947)
        at android.support.v4.app.FragmentManagerImpl.attachFragment(FragmentManager.java:1280)
        at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:672)
        at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1467)
        at android.support.v4.app.FragmentManagerImpl.executePendingTransactions(FragmentManager.java:472)
        at android.support.v4.app.FragmentPagerAdapter.finishUpdate(FragmentPagerAdapter.java:141)
        at android.support.v4.view.ViewPager.populate(ViewPager.java:1068)
        at android.support.v4.view.ViewPager.populate(ViewPager.java:914)
        at android.support.v4.view.ViewPager$3.run(ViewPager.java:244)
        at android.support.v4.view.ViewPager.completeScroll(ViewPager.java:1761)
        at android.support.v4.view.ViewPager.onInterceptTouchEvent(ViewPager.java:1896)
        at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1854)
        at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2211)
        at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1912)
        at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2211)
        at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1912)
        at android.view.ViewGroup.dispatchTransformedTouchEvent(ViewGroup.java:2211)
        at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:1912)
        at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:2228)
        at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1471)
        at android.app.Activity.dispatchTouchEvent(Activity.java:2424)
        at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:2176)
        at android.view.View.dispatchPointerEvent(View.java:7571)
        at android.view.ViewRootImpl$ViewPostImeInputStage.processPointerEvent(ViewRootImpl.java:3883)
        at android.view.ViewRootImpl$ViewPostImeInputStage.onProcess(ViewRootImpl.java:3778)
        at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
        at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3429)
        at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3398)
        at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:3483)
        at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3406)
        at android.view.ViewRootImpl$AsyncInputStage.apply(ViewRootImpl.java:3540)
        at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
        at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3429)
        at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3398)
        at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3406)
        at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
        at android.view.ViewRootImpl.deliverInputEvent(ViewRootImpl.java:5419)
        at android.view.ViewRootImpl.doProcessInputEvents(ViewRootImpl.java:5399)
        at android.view.ViewRootImpl.enqueueInputEvent(ViewRootImpl.java:5370)
        at android.view.ViewRootImpl$WindowInputEventReceiver.onInputEvent(ViewRootImpl.java:5493)
        at android.view.InputEventReceiver.dispatchInputEvent(InputEventReceiver.java:182)
        at android.os.MessageQueue.nativePollOnce(Native Method)
        at android.os.MessageQueue.next(MessageQueue.java:132)
        at android.os.Looper.loop(Looper.java:124)
        at android.app.ActivityThread.main(ActivityThread.java:5289)
        at java.lang
Run Code Online (Sandbox Code Playgroud)

Xie*_*eyi 50

更新:

我已经阅读了FragmentManager的源代码,最后得到了真正的原因:当viewpager附加到其父级之前,片段想要附加到viewpager时会发生此异常.换句话说,在getView()方法返回之前,片段会膨胀.然后调用ViewPager容器的findViewById()方法,但ViewPager仍处于分离状态,因此找到null并抛出IllegalArgumentException.

解决方案是创建自定义ViewPager并延迟设置适配器:

public class BannerViewPager extends ViewPager {
    PagerAdapter mPagerAdapter;

    @Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        if (mPagerAdapter != null) {
            super.setAdapter(mPagerAdapter);
            mPageIndicator.setViewPager(this);
        }
    }

    @Override
    public void setAdapter(PagerAdapter adapter) {
    }

    public void storeAdapter(PagerAdapter pagerAdapter) {
        mPagerAdapter = pagerAdapter;
    }

    public BannerViewPager(Context context) {
        super(context);
    }

    public BannerViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

}
Run Code Online (Sandbox Code Playgroud)

在getView()方法中,使用storeAdapter()而不是setAdapter.

以下陈述不正确.上面的话就是实际的原因.


最后我得到了答案.它由两部分组成.

  1. 在父ViewPager中我使用a FragmentPagerAdapter来保存片段,但现在我使用了一个FragmentStatePagerAdapter.这两者之间的区别可以在这里找到:FragmentPagerAdapter和FragmentStatePagerAdapter之间的区别.
    简单来说,FragmentPagerAdapter将在片段停止时存储更多信息.在这种情况下,父ViewPager中的第一个片段会被停止但不会被销毁,而此片段中的视图将被销毁.恢复后,片段尝试重新膨胀所有视图.但是在getView()调用该方法并重新创建子ViewPager之前,子FragmentManager会尝试查找子ViewPager以保存先前存储的片段.因此,出现"java.lang.IllegalArgumentException:找不到id的视图".

  2. 用FragmentStatePagerAdapter替换FragmentPagerAdapter后,出现另一个问题.当父片段(父视图中的第一个片段)被停止,销毁和恢复时,子视图抓取器丢失.当选择第一个片段时很快就会发生这种情况,之后很快就会选择第三个片段,最后重新选择第一个片段.
    我认为这是android sdk的一个bug.灵感来自这里这里,我使用一些棘手的方法来解决问题.关键是,当一个父片段被销毁时,字段成员--- mChildFragmentManager"最终会破坏内部状态"并且不会被彻底清除.重新创建父片段时,mChildFragmentManager不为空,但子片段在被销毁之后已经被销毁,这是由mChildFragmentManager管理的.因此,子ViewPager在屏幕上显示一个空视图,该视图响应实际上不存在的假片段.有趣的是,在子ViewPager上直接滑动几次后,子片段和视图再次出现.

这是代码:

父适配器:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    if (convertView == null) {
        convertView = getBannerView(mParent);
    }
    mViewPager = (ViewPager) convertView;
    if (mBanners != null && !mBanners.isEmpty()) {
        if (mPagerAdapter == null) {
            FragmentManager childFM = mFragment.getChildFragmentManager();
            removeOldFragment(childFM);
            mPagerAdapter = new ScreenSlidePagerAdapter(childFM, mBanners);
            mViewPager.setAdapter(mPagerAdapter);
        }
    }
    return convertView;
}
Run Code Online (Sandbox Code Playgroud)

关键方法:

    private void removeOldFragment(FragmentManager fm) {
        try {
            Field added = fm.getClass().getDeclaredField("mAdded");
            added.setAccessible(true);
            added.set(fm, null);
        } catch (NoSuchFieldException e) {
            throw new RuntimeException(e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
        try {
            Field active = fm.getClass().getDeclaredField("mActive");
            active.setAccessible(true);
            active.set(fm, null);
        } catch (NoSuchFieldException e) {
            throw new RuntimeException(e);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }
Run Code Online (Sandbox Code Playgroud)

  • 我相信在onDetachedFromWindow中调用super.setAdapter(null)也需要它才能工作.我在一个选项卡Fragment中有一个ViewPager,它是另一个ViewPager中包含的许多选项卡之一,即使在内部ViewPager上使用惰性存储适配器方法,我的应用程序也会崩溃.在onDetachedFromWindow中添加super.setAdapter(null)似乎解决了这个问题. (4认同)
  • 我是一个菜鸟,但我从哪里获得变量mPageIndicator?它位于onAttachedToWindow()中的自定义ViewPager中 (3认同)
  • 这似乎对我有用:)但我不确定为什么我们必须创建一个新方法,而不是在重写的setAdapter()中做同样的工作? (2认同)

小智 8

如果你ViewPager一直在Fragment,那么使用 not getFragmentManagerbut getChildFragmentManager


zMa*_*Man 6

谢毅在解释异常背后的推理方面做得很好.

这是我找到的解决方案:

@Override
public void onPause() {
    super.onPause();

    for ( Fragment f : getChildFragmentManager().getFragments() ) {
        if ( f instanceof MyFragmentType ) {
            getChildFragmentManager().beginTransaction().remove( f ).commit();
        }
    }

}
Run Code Online (Sandbox Code Playgroud)

每当我去一个新片段然后回来时,我都会遇到异常.确保在"cleaup"方法中删除片段,然后加载可以在返回片段时重新加载视图.


Sky*_*ker 5

当您有一个Activity和一个Activity会包含的Fragment时,您将拥有具有ID的layout-xml,应该在其中放置脆弱部分。

如果id不在“活动”中,则将获得此异常。

例:

getSupportFragmentManager().beginTransaction()
            .add(R.id.banner_container, bannerFragment)
            .commit();
Run Code Online (Sandbox Code Playgroud)

如果R.id.banner_container不是Activity-layout-xml中的id,则将获得异常。


Sim*_*aes 5

使用嵌套的ViewPagers时遇到同样的问题.我通过更换这两个固定的问题FragmentPagerAdapter以s FragmentStatePagerAdapter秒.


Jag*_*eon 5

虽然谢一的解决方案工作得很好,但它几乎是基于进行反射,这并不是一件好事。

作为替代答案,我设法使用片段事务删除了子片段,我认为这应该是此处的首选方法。

首先,向子片段管理器注册片段生命周期回调,以便我们可以保持对子片段的引用。

private List<Fragment> mFragments = new ArrayList<>();

@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);

    …(DO YOUR STUFF)
    //register a fragment lifecycle callback
    getChildFragmentManager().registerFragmentLifecycleCallbacks(new FragmentManager.FragmentLifecycleCallbacks() {

        @Override
        public void onFragmentCreated(FragmentManager fm, Fragment f, Bundle savedInstanceState) {
            super.onFragmentCreated(fm, f, savedInstanceState);
            mFragments.add(f);
        }

        @Override
        public void onFragmentDestroyed(FragmentManager fm, Fragment f) {
            super.onFragmentDestroyed(fm, f);
            mFragments.remove(f);
        }
    }, false);
Run Code Online (Sandbox Code Playgroud)

接下来,我们只需要遍历片段列表以将它们从 onCreateView 中的子片段管理器中删除:

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
                         @Nullable Bundle savedInstanceState) {
    //remove child fragments before inflating view
    Iterator<Fragment> iterator = mFragments.iterator();
    while (iterator.hasNext()) {
        Fragment fragment = iterator.next();
        iterator.remove();
        getChildFragmentManager().beginTransaction()
                .remove(fragment)
                .commitNow();
    }
    return inflater.inflate(R.layout.your_fragment, container, false);
}
Run Code Online (Sandbox Code Playgroud)

就是这样。