9

我正在设计一个允许用户在 ViewPager 中的多个页面之间切换的应用程序。我一直在努力弄清楚如何在屏幕上不再可见时从页面中删除 Fragment 实例,将其缓存(例如,HashMap),然后将其还原,以便当用户翻回该页面,其中的视图和其他所有内容都将处于与删除之前相同的状态。例如,我的第一个页面是一个登录屏幕,它使该特定页面上的某些布局元素在成功登录时可见/不可见。当我向前翻了足够多的页面然后翻回第一页时,布局被重置。对于我的另一个页面来说,这成为一个更大的问题,其中包含一个巨大的水平/垂直滚动数据网格,我在初始化时使用后台线程进行绘制。

所以我做了一些研究...

我浏览了 FragmentStatePageAdapter 的源代码,在 destroyItem() 回调中,被删除的 Fragment 实例的状态被保存到一个 ArrayList 中。当在 instantiateItem() 回调中创建 Fragment 的新实例时,如果项目的实例尚不存在(它们通过使用 ArrayList 来跟踪这一点),则会创建一个新的 Fragment 实例并保存其状态使用相应的 Fragment.SavedState 数据进行初始化。不幸的是,这些数据不包括视图所处的状态,尽管我注意到对于具有 GridView/ListView 的页面,视图的状态以某种方式恢复(如果我滚动到某个随机位置,翻了几页然后回来,它不会被重置)。

根据 API:

保存的状态不能包含对其他片段的依赖——也就是说,它不能使用 putFragment(Bundle, String, Fr​​agment) 来存储片段引用,因为以后使用这个保存的状态时该引用可能无效。同样,Fragment 的目标和结果代码也不包含在此状态中。

作为 Android 的菜鸟,我不太确定我是否理解最后一句话。

话虽这么说,有没有办法缓存视图状态?如果没有,我想我会继续将所有片段页面留在内存中。

4

5 回答 5

4

我有同样的问题问题并通过实现这两个功能解决了

    public void onSaveInstanceState (Bundle outState)
    public void onActivityCreated (Bundle savedInstanceState)

在我想保存的片段上。在第一个函数中,您应该在 Bundle 中保存恢复视图所需的日期(在我的情况下,我有一堆微调器,所以我使用了一个 int 数组来保存它们的位置)。第二个函数在恢复片段时调用,是您实现恢复过程的地方。

我希望这有帮助。我还让我的适配器从 FragmentStatePageAdapter 继承,但我不确定这是强制性的。

于 2011-08-11T17:30:23.520 回答
2

main.xml 列表

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <TextView android:text="Page 1" android:id="@+id/textViewHeader"
        android:layout_width="fill_parent" android:layout_height="wrap_content"
        android:gravity="center" android:padding="10dip" android:textStyle="bold"></TextView>
    <android.support.v4.view.ViewPager
        android:layout_width="fill_parent" android:layout_height="fill_parent"
        android:id="@+id/viewPager" />
</LinearLayout>

设置 ViewPager

ViewPager viewPager = (ViewPager) findViewById(R.id.viewPager);
MyPagerAdapter adapter = new MyPagerAdapter(this);
viewPager.setAdapter(adapter);

寻呼机适配器

@Override
public void destroyItem(View view, int arg1, Object object) {
         ((ViewPager) view).removeView((View)object);
}
@Override
public int getCount() {
          return views.size();
}
@Override
public Object instantiateItem(View view, int position) {
           View view = views.get(position);
           ((ViewPager) view).addView(view);
           return view;
}
@Override
public boolean isViewFromObject(View view, Object object) {
           return view == object;
}

在这里查看更多详细信息查看寻呼机示例

于 2011-11-02T16:03:23.247 回答
1

查看各种文档片段,我最好的猜测是您正在创建的视图没有附加 ID。假设片段的保存状态是从 Fragment.onSaveInstanceState 创建的,那么片段将自动保存任何具有 id 的视图状态。如果您从布局文件创建它们,您可能有一个与您的 ListView/GridView 关联的默认 ID。您还可以通过调用 setId 将 id 与视图相关联。

此外,对于您的自定义填充片段,您可能还必须在 onSaveInstanceState 中执行一些自定义操作。

于 2011-08-05T13:48:40.523 回答
0

当我使用 PagerSlidingTabStrip 并使用 FragmentPagerAdapter 的实例时,我也遇到了这个问题,切换到 FragmentStatePagerAdapter 肯定有效。

然后我使用 onSaveInstanceState() 来保存状态

于 2015-04-10T14:46:22.130 回答
0

这是我如何在 PagerAdapter 中实现缓存的示例。填充缓存后,所有未来的视图请求都从缓存中提供,只有数据被替换。

public class TestPageAdapter extends PagerAdapter{

private int MAX_SIZE = 3;
private ArrayList<SoftReference<View>> pageCache = new ArrayList<SoftReference<View>>(3);


public TestPageAdapter(Context context){
    // do some initialization
}

@Override
public int getCount() {
    // number of pages
}

private void addToCache(View view){
    if (pageCache.size() < MAX_SIZE){
        pageCache.add(new SoftReference<View>(view));
    } else {
        for(int n = (pageCache.size()-1); n >= 0; n--) {
            SoftReference<View> cachedView = pageCache.get(n);
            if (cachedView.get() == null){
                pageCache.set(n, new SoftReference<View>(view));
                return;
            }
        }
    }
}

private View fetchFromCache(){
    for(int n = (pageCache.size()-1);  n>= 0; n--) {
        SoftReference<View> reference = pageCache.remove(n);
        View view = reference.get();
        if (view != null) {
            return view;
        }
    }
    return null;
}

@Override
public Object instantiateItem(View collection, int position) {
    View view = fetchFromCache();
    if (view == null) {
        // not in cache, inflate manually
        LayoutInflater inflater = (LayoutInflater) collection.getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        view = inflater.inflate(R.layout.page, null);
    } 
    setData(view, position);
    ((ViewPager) collection).addView(view, 0);
    return view;         
}

private void setData(View view, int position){
    // set page data (images, text ....)
}

public void setPrimaryItem(ViewGroup container, int position, Object object) {
    currentItem = (View)object;
}

public View getCurrentItem() {
    return currentItem;
}

@Override
public boolean isViewFromObject(View view, Object object) {
    return view == ((View) object);
}

@Override
public void destroyItem(View collection, int arg1, Object view) {
    ((ViewPager) collection).removeView((View) view);
    addToCache((View) view);
}

}
于 2012-10-03T08:58:23.690 回答