ViewPager Fragment 被调用两次

5

好的,所以我正在尝试使用ViewPager实现三个页面,并在每个页面上具有独特的内容,因此我设置了一个系统,根据页面索引来填充不同的布局。只要我导航,这就可以工作,但我注意到Fragment被调用了两次。

这是我的FragmentAcivity类:

public class TutorialScreen extends FragmentActivity {

    private static final String TAG = "TAG";
    private static final int NUM_PAGES = 3;
    private ViewPager pager; // animation handler
    private PagerAdapter adapter; // provides pages to pager

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.tutorial_layout);

        // Instantiate Pager & Adapter
        pager = (ViewPager) findViewById(R.id.pager);
        adapter = new ScreenSlidePagerAdapter(getSupportFragmentManager());
        pager.setAdapter(adapter);
        Log.v(TAG, "Initial Page is " + pager.getCurrentItem()); // this logs '0' - which is what i want
        pager.setCurrentItem(pager.getCurrentItem());
    }

    @Override
    public void onBackPressed() {
        if(pager.getCurrentItem() == 0) {
            super.onBackPressed();
        }
        else {
            pager.setCurrentItem(pager.getCurrentItem() -1);
        }
    }

    // Pager Adapter SubClass
    private class ScreenSlidePagerAdapter extends FragmentStatePagerAdapter {

        public ScreenSlidePagerAdapter(FragmentManager fm) {
            super(fm);
            Log.v(TAG, "constructor"); // this logs once
            pager.setCurrentItem(pager.getCurrentItem());
        }

        @Override
        public Fragment getItem(int position) {
            Log.v(TAG, "getItem " + position); // this logs twice
            return new ScreenSlidePageFragment(position); // note I've overridden the constructor
        }

        @Override
        public int getCount() {
            return NUM_PAGES;
        }   
    } 
}

还有我的 Fragment 类:

public class ScreenSlidePageFragment extends Fragment {

    private static int pageIndex;
    private static final String TAG = "TAG";

    ScreenSlidePageFragment(int id) {
        pageIndex = id;
        Log.v(TAG, "Page Index is " + pageIndex); // this logs twice
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

        int resID = -1;

        switch(pageIndex) {
            case 0:
                resID = R.layout.tutorial_page1;
                break;
            case 1:
                resID = R.layout.tutorial_page2;
                break;
            case 2:
                resID = R.layout.tutorial_page3;
                break;
            default:
                resID = R.layout.tutorial_page1;
                break;
        }

        Log.v(TAG, "Res Index is " + resID); // this logs twice
        ViewGroup rootView = (ViewGroup) inflater.inflate(resID, container, false);
        return rootView;
    }
}

日志是

构造函数 getItem 0 页面索引为0 getItem 1 页面索引为1 资源索引为2130903043 资源索引为2130903043

因此无论我是否使用setCurrentItem()设置位置,它基本上都会立即跳转到第二页...

我认为我可能在尝试只拥有一个Fragment类而不是每个匹配布局的单独类时作弊,但是...这应该可以工作吧?

请注意,我还遇到了一个(我认为)相关的问题,即第二个(最初的)页面会显示两次 - 首先是初始页面,然后是再次向右滑动到下一页后(应该是第3页?)。一旦我导航到第3页(经过两次滑动),我就可以一路回到第1页(应该是最初的页面)。

你有什么想法?我做错了什么?


1
默认情况下,左侧和右侧的片段(最初只有一个在右侧)会被加载。这是出于性能考虑,以便在分页时可以实现平滑的动画效果。您可以增加每侧预加载的数量,但最少为1。 - Dan Harms
好的,那有道理(我记得在 transformPage() 方法中看到过这个)...但它为什么要导航/设置第二页?我应该在其他地方调用 setCurrentItem() 吗? - jesses.co.tt
根据你的日志,看起来它正在创建页面0和页面1(在后台)。是什么让你觉得页面1被显示了两次?另外,pager.setCurrentItem(pager.getCurrentItem()); 这行代码没有任何作用,所以你可以将其删除。 - Dan Harms
你尝试在 pager.setCurrentItem 语句中明确放置0了吗? - Dan Harms
我有看到这个在几个答案里提到,但它不会改变日志或屏幕结果中的任何东西... :/ - jesses.co.tt
显示剩余2条评论
3个回答

3
因此,我解决了自己的(根)问题(尽管没有解决我遇到的奇怪的索引情况)。
我对在ViewPager中为每个页面实例化单独内容的理解是,会重复使用一个单一的Fragment,并且我将通过向其构造函数传递参数来告诉它要膨胀哪个布局。
相反,我意识到我应该拥有完全独立的Fragment类,每个类都拥有自己独特的布局,并且我可以使用getItem()中的相同位置id在FragmentStatePagerAdapter中实例化正确的Fragment...现在对我来说很合理。
请参见下面的代码:
public class TutorialScreen extends FragmentActivity {

    private static final int NUM_PAGES = 3;
    private ViewPager pager; 
    private PagerAdapter adapter; 

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.tutorial_layout);

        // Instantiate Pager & Adapter
        pager = (ViewPager) findViewById(R.id.pager);
        adapter = new ScreenSlidePagerAdapter(getSupportFragmentManager());
        pager.setAdapter(adapter);
    }

    @Override
    public void onBackPressed() {
        if(pager.getCurrentItem() == 0) {
            super.onBackPressed();
        }
        else {
            pager.setCurrentItem(pager.getCurrentItem() -1);
        }
    }

    // PagerAdapter SubClass
    private class ScreenSlidePagerAdapter extends FragmentStatePagerAdapter {

        public ScreenSlidePagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            Fragment fragment;
            switch(position) {
                case 0:
                    fragment = new TutorialPage1();
                break;
                case 1:
                    fragment = new TutorialPage2();
                break;
                case 2:
                    fragment = new TutorialPage3();
                break;
                default:
                    fragment = new TutorialPage1();
                break;
            }
            return fragment;
        }

        @Override
        public int getItemPosition(Object object) {
            return POSITION_NONE;
        }

        @Override
        public int getCount() {
            return NUM_PAGES;
        }

    } /* EOC */

} /* EOC */ 

1
你的代码中有一个片段,在每次加载和旋转手机时都被调用了两次,你能修复一下吗? - mahdi pishguy
嗯...是的,但我认为这超出了原始问题的范围... - jesses.co.tt
你能帮我看一下代码吗?我遇到了一个问题,但是我无法解决它。 - mahdi pishguy
发布一个新问题并在此处链接。 - jesses.co.tt
但是如果你有二十个不同的片段,而你只需要在每个片段上改变一些东西,你真的想创建二十个不同的布局吗? - Shikhar Mainalee
不,你不会这样做,你需要一些类型的生成器来将你的模型中的值注入/绑定到你的视图中。 - jesses.co.tt

0

mViewPager.setOffscreenPageLimit(1); 将数字1替换为片段的数量


0

所以,我正在使用相同的片段。 适配器:

public class IncomeWeekFragmentAdapter extends FragmentStatePagerAdapter {

    List<String> titlesList;
    List<String> reqDates;

    public IncomeWeekFragmentAdapter(FragmentManager fm, List<String> titles, List<String> reqDates) {
        super(fm);
        this.titlesList = titles;
        this.reqDates = reqDates;
    }

    @Override
    public Fragment getItem(int position) {
        Fragment fragment = null;
        Bundle args = new Bundle();
        if(fragment==null)
        fragment = new IncomeWeekFragment();
        if (position == 0) {
            args.putString("date_start", reqDates.get(1));
            args.putString("date_end", reqDates.get(0));
        } else if (position == 1) {
            args.putString("date_start", reqDates.get(3));
            args.putString("date_end", reqDates.get(2));
        } else if (position == 2) {
            args.putString("date_start", reqDates.get(5));
            args.putString("date_end", reqDates.get(4));
        } else if (position == 3) {
            args.putString("date_start", reqDates.get(7));
            args.putString("date_end", reqDates.get(6));
        }
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public int getCount() {
        return COUNT_INCOME_FRAGMENT;
    }

    @Override
    public int getItemPosition(@NonNull Object object) {
        return POSITION_NONE;
    }

    @Nullable
    @Override
    public CharSequence getPageTitle(int position) {
        return titlesList.get(position);
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接