android 停止FragmentPagerAdapter一次创建其所有片段

eeq64g8w  于 2024-01-04  发布在  Android
关注(0)|答案(3)|浏览(164)

所以我有一个底部导航栏,每个标签有4个片段,在每个片段中,我调用一个API请求来获取一些数据,但问题是每次我按下栏的任何标签,至少有两个片段被创建,它们调用自己的方法,并通过扩展触发API请求..!我只希望我选择的片段被示例化。
我知道适配器这样做是为了预先呈现片段,以确保在选项卡和诸如此类的东西之间进行更好的事务处理..!但我真的不能在每次选择时调用多个API调用..!

适配器

  1. public class My_PagerAdapter extends FragmentPagerAdapter {
  2. // I've tried FragmentStatePagerAdapter but same thing
  3. public My_PagerAdapter (FragmentManager fm) {
  4. super(fm);
  5. }
  6. @Override
  7. public Fragment getItem(int position) {
  8. switch (position) {
  9. case 0:
  10. new MyFragment_1();
  11. case 1:
  12. new MyFragment_2();
  13. case 2:
  14. new MyFragment_3();
  15. case 3:
  16. new MyFragment_4();
  17. }
  18. }
  19. @Override
  20. public int getCount() {
  21. return 4;
  22. }
  23. }

字符串

编辑

这就是我所谓的适配器。

  1. ViewPager viewPager = main.findViewById(R.id.vp);
  2. viewPager.setOffscreenPageLimit(1);
  3. viewPager.setAdapter(new My_PagerAdapter (getChildFragmentManager()));
  4. navigationTabBar.setModels(models); // just UI stuff for each tab offered by the bottom navigation bar library,
  5. navigationTabBar.setViewPager(viewPager);

a11xaf1n

a11xaf1n1#

我在做一个项目的时候也遇到了同样的问题
我的解决方案是在每个片段中添加OnResume方法上的API调用。这样,只有当片段完全可见时,它们才会被触发。
查看fragment lifecycle

yhxst69z

yhxst69z2#

好吧,这正是我面临的一个问题。我的解决方案并没有阻止viewpager创建片段,但它会停止对网络API的调用。
要点如下:
1)创建一个接口

  1. public interface ViewPagerLifeCycleManagerInterface {
  2. void onResumeAndShowFragment();
  3. void onPauseAndHideFragment();
  4. //void refreshFragment();
  5. }

字符串
2)修改FragmentPagerAdapter以覆盖onInstantiateItem方法
在这里,每个Fragment都将在Adapter类中声明一个weakReference,以便存储对所创建的Fragment的引用

  1. @Override
  2. public Object instantiateItem(ViewGroup container, int position){
  3. Fragment createdFragment = (Fragment) super.instantiateItem(container, position);
  4. switch (position){
  5. case 0:
  6. xyzWeakReference=null;
  7. xyzFragmentWeakReference=new WeakReference<>((xyz)createdFragment);
  8. break;
  9. case 1:
  10. xyz1WeakReference=null;
  11. xyz1WeakReference=new WeakReference<>((xyz1WeakReference)createdFragment);
  12. break;
  13. }
  14. return createdFragment;
  15. };


3)在FragmentPagerAdapter中,添加以下方法以获取图片中片段的弱引用

  1. public Fragment getFragmentAtGivenPosition(int i){
  2. switch (i){
  3. case 0:
  4. if(xyzFragmentWeakReference == null){
  5. return null;
  6. }
  7. return xyzFragmentWeakReference.get();
  8. case 1:
  9. if(xyz1FragmentWeakReference == null){
  10. return null;
  11. }
  12. return xyz1FragmentWeakReference.get();
  13. }
  14. }


4)现在,在创建TabLayout并示例化视图页导航的Activity中,将侦听器附加到TabLayout以侦听选项卡更改

  1. tabLayout_bookmarks.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
  2. @Override
  3. public void onTabSelected(final TabLayout.Tab tab) {
  4. //let the instantiateItem have some time to be called by the adapter
  5. currentFragmentIndex = tab.getPosition();
  6. new Handler().postDelayed(new Runnable() {
  7. @Override
  8. public void run() {
  9. ViewPagerLifeCycleManagerInterface currentFragment = (ViewPagerLifeCycleManagerInterface)btca.getFragmentAtGivenPosition(tab.getPosition());
  10. if(currentFragment!=null){
  11. currentFragment.onResumeAndShowFragment();
  12. }else{
  13. //Log.d("FragmentCreate","Current fragment is null and fucked up in adapter");
  14. //if it is null ... that means the adapter hasn't yet called instantiate item ... this internally calls get item any way
  15. //.....
  16. //This shouldn't really hit but in case it does ... keep a handler in order to ensure that everything is created
  17. new Handler().postDelayed(new Runnable() {
  18. @Override
  19. public void run() {
  20. ViewPagerLifeCycleManagerInterface localFragment = (ViewPagerLifeCycleManagerInterface)btca.getItem(tab.getPosition());
  21. //getItem never returns a null fragment unless supplied a horrendous value for position
  22. //by the time these 50 ms pass, the instantiate item should surely have been called
  23. //else it will be an empty space ... no crash though
  24. localFragment.onResumeAndShowFragment();
  25. }
  26. },50);
  27. }
  28. }
  29. },100);
  30. }
  31. @Override
  32. public void onTabUnselected(final TabLayout.Tab tab) {
  33. new Handler().postDelayed(new Runnable() {
  34. @Override
  35. public void run() {
  36. ViewPagerLifeCycleManagerInterface currentFragment = (ViewPagerLifeCycleManagerInterface)btca.getFragmentAtGivenPosition(tab.getPosition());
  37. if(currentFragment!=null){
  38. currentFragment.onPauseAndHideFragment();
  39. }else{
  40. //Log.d("FragmentCreateTab","the fucking fragment was null");
  41. //if it is null ... that means the adapter hasn't yet called instantiate item ... this internally calls get item any way
  42. //.....
  43. //This shouldn't really hit but in case it does ... keep a handler in order to ensure that everything is created
  44. new Handler().postDelayed(new Runnable() {
  45. @Override
  46. public void run() {
  47. ViewPagerLifeCycleManagerInterface localFragment = (ViewPagerLifeCycleManagerInterface)btca.getItem(tab.getPosition());
  48. //getItem never returns a null fragment unless supplied a horrendous value for position
  49. //by the time these 50 ms pass, the instantiate item should surely have been called
  50. //else it will be an empty space ... no crash though
  51. localFragment.onPauseAndHideFragment();
  52. }
  53. },50);
  54. }
  55. }
  56. },100);
  57. }
  58. @Override
  59. public void onTabReselected(TabLayout.Tab tab) {
  60. //do nothing
  61. }
  62. });


5)在Viewpager中的每个Fragment中,实现我们在步骤1中创建的Interface并覆盖方法。
在每个片段中创建一个布尔变量amIVisible.这将有助于决定片段何时可见以及何时可以调用网络API
a)这里对于viewpager中的第一个片段,即索引为0的片段,网络API调用必须在视图创建后立即发生。默认情况下,该片段显然是可见的。这是在oncodeView方法中编写的

  1. if(dataList!=null && dataList.size()==0) {
  2. if (savedInstanceState==null) {
  3. //your api call to load from net
  4. } else {
  5. if (savedInstanceState.getBoolean("savedState")) {
  6. //If you have saved data in state save, load it here
  7. } else {
  8. //only fire the async if the current fragment is the one visible, else the onResumeAndShowFragment will trigger the same async when it becomes visible
  9. if (savedInstanceState.getBoolean("amIVisible")) {
  10. //Load data from net
  11. }
  12. }
  13. }
  14. }


对于第一个片段,其他方法如下所示

  1. @Override
  2. public void onResumeAndShowFragment() {
  3. amIVisible=true;
  4. if(dataList!=null && dataList.size()==0){
  5. new Handler().postDelayed(new Runnable() {
  6. @Override
  7. public void run() {
  8. //Load data from net if data was not found,
  9. //This basically means auto refresh when user scrolls back and the fragment had no data
  10. }
  11. },400);
  12. }
  13. }
  14. @Override
  15. public void onPauseAndHideFragment() {
  16. amIVisible=false;
  17. }


在这里,我已经覆盖了onSaveInstanceState方法,并保存了amIVisible的值,savedState是一个布尔值,表示列表是否至少有1个项目。
B)对于其他片段,数据将通过以下过程加载

  1. if(savedInstance!=null){
  2. if (savedInstance.getBoolean("savedState")) {
  3. new Handler().postDelayed(new Runnable() {
  4. @Override
  5. public void run() {
  6. //load data from saved State
  7. }
  8. },100);
  9. } else {
  10. //only fire the async if the current fragment is the one visible, else the onResumeAndShowFragment will trigger the same async when it becomes visible
  11. if (savedInstance.getBoolean("amIVisible")) {
  12. new Handler().postDelayed(new Runnable() {
  13. @Override
  14. public void run() {
  15. //load data from net
  16. }
  17. },100);
  18. }
  19. }
  20. }


其他片段的接口方法相同。
这是相当复杂的,但做的工作。适配器内的弱引用甚至允许垃圾收集和避免上下文泄漏。

展开查看全部
ca1c2owp

ca1c2owp3#

防止加载视图页导航中的所有片段。
在视图分页器的setAdapret前添加此行. viewPager.setOffscreenPageLimit(number_of_fragment); //示例:4
并检查登录onResume。可以更新其中的数据

相关问题