当我在 TabLayout 中有 3 个选项卡时,如何理解 getFragments.size() 给我 2,每个选项卡都显示正确的片段?
How to understand getFragments.size() giving me 2 when I've got 3 tabs in TabLayout each showing the right fragment?
我有 TabLayout,其中包含使用 FragmentPagerAdapter 和 3 个片段通过 ViewPager 创建的 3 个选项卡。滑动到第三个选项卡,中断第一个选项卡的一位并更改片段的索引。
我有 TabLayout,其中包含使用 FragmentPagerAdapter 和 3 个片段通过 ViewPager 创建的 3 个选项卡。片段没问题,activity 在(重新)开始时正确显示所有内容,但我注意到一个错误,当我滑动到第三个选项卡,然后我滑动回到第一个选项卡,第一个选项卡丢失一些信息。当我重新启动 activity 时,第一个选项卡会返回正确的信息。但是每次我去第三个时都会丢失它,然后回到 first.I 应用了一个解决方法,它刷新了片段的 onResume 中丢失的位,这工作正常。但我仍然对此感到困惑,并试图找出发生这种情况的原因。
我已经检查了网络上的各种来源,并多次检查了我的代码,但我看不出是什么导致了这个问题。
然而,我注意到 tabs.getTabCount() 在我的 运行 中显示 3,但 FragmentManager getFragments.size() 令人惊讶地显示 2,当所有三个选项卡和片段都在 activity 中可见。它还显示当我滑动到第三个选项卡时片段的索引发生了变化:我希望 tab1 为 0,tab2 为 1,tab3 为 2,tab1 和 tab2 就是这种情况(tab3 不可见通过 FragmentManager)在我滑动到 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 并重新编制索引。
如果有人可以分享他们对这个问题或类似问题的经验(如果有的话),或者任何关于如何查明这个问题的想法,我将不胜感激。
基本上会发生这种情况,因为适配器仅在片段邻居上持有强引用。
当您在第一个片段上时,它会存储 1,2 个片段。
当你在第三个片段上时,它存储了3,2个片段。
所以,如果你在中间片段上调用 getFragments()
,90% 这个方法会 return 你 3.
以防万一有人遇到与我类似的情况:
samaromku 提供了我上述问题的答案,但那是我的问题的一部分。第二部分是当我滑动到 tab3 时,我正在丢失 tab1 中的数据。
我找到了一个解决方法,但它不是一个合适的解决方案。现在,我知道了答案:这是因为当显示 tab3 时适配器不再引用 tab1,当我从 tab3 滑回 tab1 时,适配器提供了一个新的 tab1 实例,然后部分由代码填充片段在 Create 上,但其余数据(例如微调器值等)丢失了。
我已经通过设置解决了这个问题:viewPager.setOffscreenPageLimit(3);现在适配器将片段保存在内存中。
有关详细信息,请参阅此内容:
我有 TabLayout,其中包含使用 FragmentPagerAdapter 和 3 个片段通过 ViewPager 创建的 3 个选项卡。滑动到第三个选项卡,中断第一个选项卡的一位并更改片段的索引。
我有 TabLayout,其中包含使用 FragmentPagerAdapter 和 3 个片段通过 ViewPager 创建的 3 个选项卡。片段没问题,activity 在(重新)开始时正确显示所有内容,但我注意到一个错误,当我滑动到第三个选项卡,然后我滑动回到第一个选项卡,第一个选项卡丢失一些信息。当我重新启动 activity 时,第一个选项卡会返回正确的信息。但是每次我去第三个时都会丢失它,然后回到 first.I 应用了一个解决方法,它刷新了片段的 onResume 中丢失的位,这工作正常。但我仍然对此感到困惑,并试图找出发生这种情况的原因。
我已经检查了网络上的各种来源,并多次检查了我的代码,但我看不出是什么导致了这个问题。 然而,我注意到 tabs.getTabCount() 在我的 运行 中显示 3,但 FragmentManager getFragments.size() 令人惊讶地显示 2,当所有三个选项卡和片段都在 activity 中可见。它还显示当我滑动到第三个选项卡时片段的索引发生了变化:我希望 tab1 为 0,tab2 为 1,tab3 为 2,tab1 和 tab2 就是这种情况(tab3 不可见通过 FragmentManager)在我滑动到 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 并重新编制索引。
如果有人可以分享他们对这个问题或类似问题的经验(如果有的话),或者任何关于如何查明这个问题的想法,我将不胜感激。
基本上会发生这种情况,因为适配器仅在片段邻居上持有强引用。
当您在第一个片段上时,它会存储 1,2 个片段。
当你在第三个片段上时,它存储了3,2个片段。
所以,如果你在中间片段上调用 getFragments()
,90% 这个方法会 return 你 3.
以防万一有人遇到与我类似的情况:
samaromku 提供了我上述问题的答案,但那是我的问题的一部分。第二部分是当我滑动到 tab3 时,我正在丢失 tab1 中的数据。
我找到了一个解决方法,但它不是一个合适的解决方案。现在,我知道了答案:这是因为当显示 tab3 时适配器不再引用 tab1,当我从 tab3 滑回 tab1 时,适配器提供了一个新的 tab1 实例,然后部分由代码填充片段在 Create 上,但其余数据(例如微调器值等)丢失了。
我已经通过设置解决了这个问题:viewPager.setOffscreenPageLimit(3);现在适配器将片段保存在内存中。
有关详细信息,请参阅此内容: