IllegalStateException: приложение PagerAdapter изменило содержимое адаптера без вызова PagerAdapter # notifyDataSetChanged

Я использую пример ViewPager с вкладками ActionBar, взятыми из документации по Android здесь.

К сожалению, как только я вызываю метод addTab, приложение вылетает со следующим исключением:

IllegalStateException: приложение PagerAdapter изменило содержимое адаптера без вызова PagerAdapter # notifyDataSetChanged! Ожидаемое количество элементов адаптера 0, найдено 1.

Это код FragmentPagerAdapter:

public static class TabsAdapter extends FragmentPagerAdapter
            implements ActionBar.TabListener, ViewPager.OnPageChangeListener {
        private final Context mContext;
        private final ActionBar mActionBar;
        private final ViewPager mViewPager;
        private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();

        static final class TabInfo {
            private final Class<?> clss;
            private final Bundle args;

            TabInfo(Class<?> _class, Bundle _args) {
                clss = _class;
                args = _args;
            }
        }

        public TabsAdapter(Activity activity, ViewPager pager) {
            super(activity.getFragmentManager());
            mContext = activity;
            mActionBar = activity.getActionBar();
            mViewPager = pager;
            mViewPager.setAdapter(this);
            mViewPager.setOnPageChangeListener(this);
        }

        public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
            TabInfo info = new TabInfo(clss, args);
            tab.setTag(info);
            tab.setTabListener(this);
            mTabs.add(info);
            mActionBar.addTab(tab);
            notifyDataSetChanged();
        }

        @Override
        public int getCount() {
            return mTabs.size();
        }

        @Override
        public Fragment getItem(int position) {
            TabInfo info = mTabs.get(position);
            return Fragment.instantiate(mContext, info.clss.getName(), info.args);
        }

        @Override
        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        }

        @Override
        public void onPageSelected(int position) {
            mActionBar.setSelectedNavigationItem(position);
        }

        @Override
        public void onPageScrollStateChanged(int state) {
        }

        @Override
        public void onTabSelected(Tab tab, FragmentTransaction ft) {
            Object tag = tab.getTag();
            for (int i=0; i<mTabs.size(); i++) {
                if (mTabs.get(i) == tag) {
                    mViewPager.setCurrentItem(i);
                }
            }
        }

        @Override
        public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        }

        @Override
        public void onTabReselected(Tab tab, FragmentTransaction ft) {
        }
    }
}

Я не изменяю свой адаптер в любой другой части моего кода, и я вызываю метод addTab из основного потока, метод addTab заканчивается вызовом notifyDataSetChanged. Как рекомендует документация:

PagerAdapter поддерживает изменения набора данных. Изменения набора данных должны основной поток и должен завершиться вызовом notifyDataSetChanged() аналогично адаптерам адаптера, полученным из BaseAdapter.

Ответ 1

Мне было трудно заставить мой ViewPager работать. В конце концов, кажется, что пример в документации неверен.

Метод addTab должен быть следующим:

public void addTab(Tab tab, Class<?> clss, Bundle args) {
        TabInfo info = new TabInfo(clss, args);
        tab.setTag(info);
        tab.setTabListener(this);
        mTabs.add(info);
        notifyDataSetChanged();
        mActionBar.addTab(tab);
    }

Обратите внимание на порядок последних трех операций. В исходном примере notifyDataSetChanged вызывался после функции mActionBar.addTab.

К сожалению, как только вы вызываете addTab на addTab ActionBar, автоматически добавляется первая добавленная вами вкладка. Из-за этого onTabSelected событие onTabSelected и при попытке извлечь страницу оно выдает IllegalStateException поскольку оно замечает расхождение между ожидаемым количеством элементов и фактическим.

Ответ 2

Я исправил его с помощью callinig notifyDataSetChanged() один раз и сразу перед следующим вызовом getCount():

private boolean doNotifyDataSetChangedOnce = false;

@Override
public int getCount() {

  if (doNotifyDataSetChangedOnce) {
    doNotifyDataSetChangedOnce = false;
    notifyDataSetChanged();
  }

  return actionBar.getTabCount();

}

private void addTab(String text) {

  doNotifyDataSetChangedOnce = true;

  Tab tab = actionBar.newTab();
  tab.setText(text);
  tab.setTabListener(this);
  actionBar.addTab(tab);

}

private void removeTab(int position) {

  doNotifyDataSetChangedOnce = true;

  actionBar.removeTabAt(position);

}

Ответ 3

Я получал эту ошибку, как вы, ссылаясь на вкладки в getCount():

@Override
    public int getCount() {
        return mTabs.size();
    }

При создании экземпляра это должно быть передано как отдельная переменная, или пейджер должен быть настроен после того, как коллекция была заполнена/мутирована.

Ответ 4

попробуйте это. Я работал

protected void onPostExecute(Void aVoid) {
            super.onPostExecute(aVoid);
            System.out.println("Asyncrona onPostExecute");
            /*::::::::::::::::::: Guardo los cambios  ::::::::::::::*/
            //menuTabs = menuTabs2;
            viewPager = (ViewPager) rootView.findViewById(R.id.viewPager_menu);
            costumAdapter = new CostumAdapter2(getActivity().getSupportFragmentManager());
            costumAdapter.notifyDataSetChanged();

            viewPager.setAdapter(costumAdapter);
            tabLayout = (TabLayout) rootView.findViewById(R.id.tablayout_menu);
            tabLayout.setOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
                @Override
                public void onTabSelected(TabLayout.Tab tab) {
                    System.out.println(" La Seleccionado: " + tab.getText()+", POSICION: "+tab.getPosition());
                    viewPager.setCurrentItem(tab.getPosition());
                }

                @Override
                public void onTabUnselected(TabLayout.Tab tab) {
                    viewPager.setCurrentItem(tab.getPosition());
                }

                @Override
                public void onTabReselected(TabLayout.Tab tab) {
                    viewPager.setCurrentItem(tab.getPosition());
                }

            });
            viewPager.setAdapter(costumAdapter);
            tabLayout.setupWithViewPager(viewPager);
            loading.closeMessage();

        }

Ответ 5

У меня была та же проблема в адаптере FragmentStatePager. in onResume() При повторном инициализации adpater и ViewPager будет показана та же ошибка IllegalStateException. Решение:

При добавлении вкладки addtab выберите вкладку, которая не будет выбрана, установив для параметра setSelected значение false.

 actionBar.addTab(tab,false);

Я думаю, что исключение появилось из-за переопределения метода onTabSelected(), такого как

 viewPager.setCurrentItem(tab.getPosition());

Поэтому при вызове addTab() он не распознал, какой viewPager и адаптер добавляет вкладку.

Ответ 6

Моя проблема была не совсем такой же, как и в аналогичном контексте.

У меня есть AlertDialog, который отображается в макете с ViewPager и использует AppCompatActivity. При повороте экрана я получал исключение IllegalStateException и сбой.

Благодаря ответам @almisoft и @Zagorax, я попробовал позвонить notifyDataSetChanged(); прямо перед тем, как я покажу диалог, и проблема, похоже, исчезла.

Ответ 7

добавьте эту строку в файл адаптера, где вызывается метод public Object instantiateItem (контейнер ViewGroup, int position)

(Контейнер (ViewPager)).addView(view); обратный просмотр;

Ответ 8

Убедитесь, что значения в setOffscreenPageLimit и getCount совпадают, в противном случае вы получите эту ошибку:

    mViewPager.setOffscreenPageLimit(MAX_TABS);

    public class SectionsPagerAdapter extends FragmentPagerAdapter {
    ...
    @Override
    public int getCount() {
        return MAX_TABS;
    }

Ответ 9

Вызовите Viewpager.setAdapter (адаптер); после добавления элемента в список массивов.

Ответ 10

Ответ One Line просто добавляет viewpageadapter.notifyDataSetChanded()