java - 当 TabLayout 中有 3 个选项卡,每个选项卡都显示正确的 fragment 时,如何理解 getFragments.size() 给我 2?

标签 java android android-fragments android-tablayout fragmentpageradapter

我有 TabLayout,其中使用 FragmentPagerAdapter 通过 ViewPager 创建了 3 个选项卡和 3 个 fragment 。滑动到第三个选项卡,中断第一个选项卡的一位并更改 fragment 的索引。

我有 TabLayout,其中使用 FragmentPagerAdapter 通过 ViewPager 创建了 3 个选项卡和 3 个 fragment 。 fragment 正常, Activity 在(重新)启动时正确显示所有内容,但我注意到一个错误,当我滑动到第三个选项卡,然后滑回第一个选项卡时,第一个选项卡会丢失一些信息。当我重新启动 Activity 时,第一个选项卡会返回正确的信息。但每次我转到第三个,然后回到第一个时,都会丢失它。我应用了一种解决方法,刷新 fragment 的 onResume 中丢失的位,并且效果很好。但我仍然对此感到困惑,并试图找出为什么会发生这种情况。

我已经检查了网络上的各种资源,并多次检查了我的代码,但我不知道是什么导致了这个问题。 然而,我注意到当我在 Activity 中运行 tabs.getTabCount() 时,它显示 3,但当所有三个选项卡和 fragment 在 Activity 中都可见时,FragmentManager getFragments.size() 令人惊讶地显示 2。它还显示,当我滑动到第三个选项卡时, fragment 的索引发生了变化:我希望 tab1 位于 0,tab2 位于 1,tab3 位于 2,tab1 和 tab2 就是这种情况(tab3 不可见)在我滑动到 tab3 之前,索引是相反的:tab1 位于 1,tab2 位于 0,tab3 仍然不在那里,尽管在 Activity 中可见。

我没有收到任何错误消息。

//I've got this in my activity
SectionsPagerAdapter sectionsPagerAdapter = new SectionsPagerAdapter(this, getSupportFragmentManager());
    ViewPager viewPager = findViewById(R.id.view_pager);
    viewPager.setAdapter(sectionsPagerAdapter);
    TabLayout tabs = findViewById(R.id.tabs);
    tabs.setupWithViewPager(viewPager);
//this is my FragmentPagerAdapter
public class SectionsPagerAdapter extends FragmentPagerAdapter {

  private static final int NUM_TABS = 3;
  @StringRes
  private static final int[] TAB_TITLES = new int[]{R.string.tab_text_1, R.string.tab_text_2, R.string.tab_text_3};
  private final Context mContext;
  private final FragmentManager fm;

  public SectionsPagerAdapter(Context context, FragmentManager fm) {
    super(fm, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT);
    mContext = context;
    this.fm = fm;

  }

  @Override
  public Fragment getItem(int position) {
    // getItem is called to instantiate the fragment for the given page.
    // Return a PlaceholderFragment (defined as a static inner class below).
    //return PlaceholderFragment.newInstance(position + 1);
    Fragment frag = null;

    switch(position){
      case 0:
        frag = new InspectionInfoFragment();
        break;
      case 1:
        frag = new TuberConditionFragment();
        break;
      case 2:
        frag = new FindingsFragment();
        break;
    }

    return frag;

  }

  @Nullable
  @Override
  public CharSequence getPageTitle(int position) {
    return mContext.getResources().getString(TAB_TITLES[position]);
  }

  @Override
  public int getCount() {
    return NUM_TABS;
  }
}
<!-- this is my activity layout -->
<?xml version="1.0" encoding="utf-8"?>
  <androidx.coordinatorlayout.widget.CoordinatorLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".view.inspection_form.InspectionFormActivity">

    <com.google.android.material.appbar.AppBarLayout
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:theme="@style/AppTheme.AppBarOverlay">

      <androidx.appcompat.widget.Toolbar
        android:id="@+id/toolbar_inspection_form"
        android:layout_width="match_parent"
        android:layout_height="?attr/actionBarSize"
        android:background="?attr/colorPrimary"
        app:popupTheme="@style/AppTheme.PopupOverlay"
          />
      <com.google.android.material.tabs.TabLayout
        android:id="@+id/tabs"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="?attr/colorPrimary"
          app:tabGravity="fill"
          app:tabMode="fixed">

      </com.google.android.material.tabs.TabLayout>

    </com.google.android.material.appbar.AppBarLayout>

    <androidx.viewpager.widget.ViewPager
      android:id="@+id/view_pager"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      app:layout_behavior="@string/appbar_scrolling_view_behavior" />

    <com.google.android.material.floatingactionbutton.FloatingActionButton
      android:id="@+id/fab"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_gravity="bottom|end"
      android:layout_margin="@dimen/fab_margin"
      app:srcCompat="@drawable/outline_assignment_black_48" />
  </androidx.coordinatorlayout.widget.CoordinatorLayout>
This is my gradle module:app

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.sasa.spcsinspections"
        minSdkVersion 19
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
        vectorDrawables.useSupportLibrary = true

        javaCompileOptions {
            annotationProcessorOptions {
                arguments = ["room.schemaLocation": "$projectDir/schemas".toString()]
            }
        }
    }
    buildTypes {
        debug {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    compileOptions {
        sourceCompatibility = '1.8'
        targetCompatibility = '1.8'
    }

    dataBinding{
        enabled true
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.google.android.material:material:1.1.0-alpha02'
    implementation 'androidx.annotation:annotation:1.0.2'
    implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
    implementation 'androidx.lifecycle:lifecycle-extensions:2.0.0'
    implementation 'androidx.preference:preference:1.1.0-alpha05'
    implementation 'androidx.legacy:legacy-support-v4:1.0.0'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test:runner:1.1.1'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
    implementation 'androidx.appcompat:appcompat:1.0.2'
    implementation 'androidx.recyclerview:recyclerview:1.1.0-alpha06'
    implementation 'com.android.support:design:28.0.0'
    implementation 'com.google.code.gson:gson:2.8.5'
    implementation 'com.fasterxml.jackson.core:jackson-databind:2.9.9.1'
    implementation "androidx.room:room-runtime:$rootProject.roomVersion"
    annotationProcessor "androidx.room:room-compiler:$rootProject.roomVersion"
    androidTestImplementation "androidx.room:room-testing:$rootProject.roomVersion"
    implementation "androidx.lifecycle:lifecycle-extensions:$rootProject.archLifecycleVersion"
    annotationProcessor "androidx.lifecycle:lifecycle-compiler:$rootProject.archLifecycleVersion"
}
// Not sure if it makes sense to include code for the three fragments + their layout files, I checked them, they are pretty straightforward and I don't see anything that would cause the problem... 

我希望 getFragments.size() 给我 3 并且不会奇怪地丢失一点 tab1 并重新索引。

如果有人可以分享他们在这个问题或类似问题上的经验(如果有的话),或者任何关于如何弄清楚这个问题的想法,我将不胜感激。

最佳答案

以防万一有人发现自己处于与我类似的情况:

samaromku 提供了我上述问题的答案,但这是我问题的一部分。第二部分是当我滑动到 tab3 时,我丢失了 tab1 中的数据。

我找到了一个解决方法,但这不是一个正确的解决方案。现在,我知道答案了:这是因为当显示 tab3 时适配器不再引用 tab1,并且当我从 tab3 刷回 tab1 时,适配器提供了 tab1 的新实例,然后由以下代码部分填充该 fragment 已创建,但其余数据(例如微调器值等)丢失了。

我已经通过设置解决了这个问题:viewPager.setOffscreenPageLimit(3);现在适配器将 fragment 保留在内存中。

有关更多详细信息,请参阅此:Android how to stop refreshing Fragments on tab change

关于java - 当 TabLayout 中有 3 个选项卡,每个选项卡都显示正确的 fragment 时,如何理解 getFragments.size() 给我 2?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57415592/

相关文章:

java - (Android+Retrofit 2) - 将静态 header 与拦截器一起使用是否有缺点?

java - ClientRequest 接受所有证书

android - 即使存在 try/catch block ,应用程序也会因空对象引用 'boolean android.content.Intent.migrateExtraStreamToClipData()' 而崩溃

android - 自定义对话框大小以匹配 Theme.Holo.Light.Dialog

android - ClassLoader 加载 android 类,它从 android 支持库扩展 android 类

android - 如何在 android 中使用 android-support-v4.jar 禁用 ViewPager 滑动

java - 在我的代码确认我删除了所有新行后,为什么我的输出仍然显示新行?

java - java中的双括号初始化

java - 错误 “duplicate entry: META-INF/MANIFEST.MF”

android - 包裹 android.os.Parcel : Unmarshalling unknown type code