前言:前几篇文章讲解了ViewPager的普通实现方法,但Android官方最推荐的一种实现方法却是使用fragment,下面我们使用fragment来重新实现一下第一篇《ViewPager 详解(一)---基本入门》所实现的效果。

系列文章:

1、《ViewPager 详解(一)---基本入门》

2、《ViewPager 详解(二)---详解四大函数》

3、《ViewPager 详解(三)---PagerTabStrip与PagerTitleStrip添加标题栏的异同》

4、《ViewPager 详解(四)----自主实现滑动指示条》

其它相关文章:

5、《Android Fragment完全解析,关于碎片你所需知道的一切》

6、《fragment中嵌套viewpager,vierpager中有多个fragment》

本篇效果图:

在第一个页面加一个Btn                                          第一页面向第二页面滑动

  

第二页面向第三个页面滑动

一、概述

从前面几篇文章,我们知道,实现ViewPager是要有适配器的,我们前面用的适配器是PagerAdapter,而对于fragment,它所使用的适配器是:FragmentPagerAdapter.先看看官方对于这个类的解释:(英文下面有中文解释)

原文:

Class Overview


Implementation of PagerAdapter that represents each page as a Fragment that is persistently kept in the fragment manager as long as the user can return to the page.

This version of the pager is best for use when there are a handful of typically more static fragments to be paged through, such as a set of tabs. The fragment of each page the user visits will be kept in memory, though its view hierarchy may be destroyed when not visible. This can result in using a significant amount of memory since fragment instances can hold on to an arbitrary amount of state. For larger sets of pages, consider FragmentStatePagerAdapter.

When using FragmentPagerAdapter the host ViewPager must have a valid ID set.

Subclasses only need to implement getItem(int) and getCount() to have a working adapter.

译文:(译的不好,大家可在评论中补充)

FragmentPagerAdapter派生自PagerAdapter,它是用来呈现Fragment页面的,这些Fragment页面会一直保存在fragment manager中,以便用户可以随时取用。

这个适配器最好用于有限个静态fragment页面的管理。尽管不可见的视图有时会被销毁,但用户所有访问过的fragment都会被保存在内存中。因此fragment实例会保存大量的各种状态,这就造成了很大的内存开销。所以如果要处理大量的页面切换,建议使用FragmentStatePagerAdapter.

每一个使用FragmentPagerAdapter的ViewPager都要有一个有效的ID集合,有效ID的集合就是Fragment的集合(感谢夫诸同学的提示)

对于FragmentPagerAdapter的派生类,只需要重写getItem(int)和getCount()就可以了。

二、具体实现

1、适配器实现——FragmentPagerAdapter

先看完整代码,再细讲:

[java] view plaincopy
  1. public class FragAdapter extends FragmentPagerAdapter {
  2. private List<Fragment> mFragments;
  3. public FragAdapter(FragmentManager fm,List<Fragment> fragments) {
  4. super(fm);
  5. // TODO Auto-generated constructor stub
  6. mFragments=fragments;
  7. }
  8. @Override
  9. public Fragment getItem(int arg0) {
  10. // TODO Auto-generated method stub
  11. return mFragments.get(arg0);
  12. }
  13. @Override
  14. public int getCount() {
  15. // TODO Auto-generated method stub
  16. return mFragments.size();
  17. }
  18. }

这里有三个函数,根据第一部分的官方文档,可知,对于FragmentPagerAdapter的派生类,只重写getItem(int)和getCount()就可以了。

对于构造函数,这里申请了一个Fragment的List对象,用于保存用于滑动的Fragment对象,并在创造函数中初始化:

[java] view plaincopy
  1. public FragAdapter(FragmentManager fm,List<Fragment> fragments) {
  2. super(fm);
  3. // TODO Auto-generated constructor stub
  4. mFragments=fragments;
  5. }

然后在getItem(int arg0)中,根据传来的参数arg0,来返回当前要显示的fragment,下面是getItem的官方解释,难度不大,不再细讲。

public abstract Fragment getItem (int position)

Return the Fragment associated with a specified position.

最后,getCount()返回用于滑动的fragment总数;

从构造函数所以看出,我们要构造Fragment的集合才行,所以下面我们就先产生我们所需要的Fragment类;

2、三个Fragment类

第一个Fragment类:

XML:(layout1.xml)

[html] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:background="#ffffff"
  6. android:orientation="vertical" >
  7. <Button android:id="@+id/fragment1_btn"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. android:text="show toast"
  11. />
  12. </LinearLayout>

在其中加入了一个Btn

Java代码:

[java] view plaincopy
  1. public class Fragment1 extends Fragment {
  2. @Override
  3. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  4. Bundle savedInstanceState) {
  5. // TODO Auto-generated method stub
  6. View view= inflater.inflate(R.layout.layout1, container, false);
  7. //对View中控件的操作方法
  8. Button btn = (Button)view.findViewById(R.id.fragment1_btn);
  9. btn.setOnClickListener(new View.OnClickListener() {
  10. @Override
  11. public void onClick(View v) {
  12. // TODO Auto-generated method stub
  13. Toast.makeText(getActivity(), "点击了第一个fragment的BTN", Toast.LENGTH_SHORT).show();
  14. }
  15. });
  16. return view;
  17. }
  18. }

在onCreateView()中返回要显示的View,上面这段代码简单演示了如何对视图里的控件进行操作,难度不大,不再细讲,如果对Fragment不太熟悉的同学,先看看这篇文章:《Android Fragment完全解析,关于碎片你所需知道的一切》

第二个Fragment类:

XML代码:(layout2.xml)原生代码,没有做任何更改

[html] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:background="#ffff00"
  6. android:orientation="vertical" >
  7. </LinearLayout>

java代码:

[java] view plaincopy
  1. public class Fragment2 extends Fragment {
  2. @Override
  3. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  4. Bundle savedInstanceState) {
  5. // TODO Auto-generated method stub
  6. View view=inflater.inflate(R.layout.layout2, container, false);
  7. return view;
  8. }
  9. }

第三个Fragment类:

XML代码:(layout3.xml)同样,原生代码,没做任何更改

[html] view plaincopy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:background="#ff00ff"
  6. android:orientation="vertical" >
  7. </LinearLayout>

java代码:

[java] view plaincopy
  1. public class Fragment3 extends Fragment {
  2. @Override
  3. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  4. Bundle savedInstanceState) {
  5. // TODO Auto-generated method stub
  6. View view=inflater.inflate(R.layout.layout3, container, false);
  7. return view;
  8. }
  9. }

3、主activity实现

核心代码:

[java] view plaincopy
  1. public class MainActivity extends FragmentActivity {
  2. @Override
  3. protected void onCreate(Bundle savedInstanceState) {
  4. super.onCreate(savedInstanceState);
  5. setContentView(R.layout.activity_main);
  6. //构造适配器
  7. List<Fragment> fragments=new ArrayList<Fragment>();
  8. fragments.add(new Fragment1());
  9. fragments.add(new Fragment2());
  10. fragments.add(new Fragment3());
  11. FragAdapter adapter = new FragAdapter(getSupportFragmentManager(), fragments);
  12. //设定适配器
  13. ViewPager vp = (ViewPager)findViewById(R.id.viewpager);
  14. vp.setAdapter(adapter);
  15. }
  16. }

首先有一个最值得注意的地方:Activity派生自FragmentActivity,其实这是有关Fragment的基础知识,只有FragmentActivity才能内嵌fragment页面,普通Activity是不行的。

这段代码主要分为两步,第一步:构造适配器;第二步:设定适配器。

先看构造适配器的过程:

[java] view plaincopy
  1. //构造适配器
  2. List<Fragment> fragments=new ArrayList<Fragment>();
  3. fragments.add(new Fragment1());
  4. fragments.add(new Fragment2());
  5. fragments.add(new Fragment3());
  6. FragAdapter adapter = new FragAdapter(getSupportFragmentManager(), fragments);

构造一个fragment列表,然后将上面的三个Fragment类对应的实例添加进去,最后生成FragAdapter实例。
至于第二步,设定适配器,没什么好讲的。

4、可能出现的问题

问题:在MainActivity中,当写到这句:fragments.add(new Fragment1()); 向Fragment列表中添加Fragement对象实例时,会提示“无法将Fragment1()转换为fragment”

解决办法 :这是因为导入包不一致,一般的问题在于:在Fragment1中导入的是android.app.Fragment, 而在这里导入类确是:android.support.v4.app.Fragment,包不同当然无法转换,统一导入为android.support.v4.app.Fragment之后就正常了.参考文章《android之cannot convert from Fragment1 to Fragment》

源码下载地址:http://download.csdn.net/detail/harvic880925/7777849

请大家尊重原创者版权,转载请标明出处:http://blog.csdn.net/harvic880925/article/details/38660861  不胜感激!

转载于:https://www.cnblogs.com/Free-Thinker/p/5480737.html

ViewPager 详解(五)-----使用Fragment实现ViewPager滑动相关推荐

  1. ViewPager 详解(三)---PagerTabStrip与PagerTitleStrip添加标题栏的异同

     相关文章: 1.<ViewPager 详解(一)---基本入门> 2.<ViewPager 详解(二)---详解四大函数> 3.<ViewPager 详解(三)-- ...

  2. ViewPager 详解(二)---详解四大函数

    前言:上篇中我们讲解了如何快速实现了一个滑动页面,但问题在于,PageAdapter必须要重写的四个函数,它们都各有什么意义,在上节的函数内部为什么要这么实现,下面我们就结合Android的API说明 ...

  3. ViewPager详解

    ViewPager详解 ViewPager中的主要方法详解 OnPageChangeListener中的三个方法详解 三种适配器的使用及其主要方法详解 ViewPager用于实现页面间的切换. Vie ...

  4. Fragment详解之四——管理Fragment(2)

    相关文章: 1.<Fragment详解之一--概述> 2.<Fragment详解之二--基本使用方法> 3.<Fragment详解之三--管理Fragment(1)> ...

  5. Masonry自动布局详解五:比例(multipliedBy)

    Masonry自动布局详解五:比例(multipliedBy) 标签: iosmasonryautolayout自动布局约束 2015-11-30 16:30 1816人阅读 评论(0) 收藏 举报 ...

  6. PackageManagerService启动详解(五)之Android包信息体和解析器(中)

        PKMS启动详解(五)之Android包信息体和包解析器(中) Android PackageManagerService系列博客目录: PKMS启动详解系列博客概要 PKMS启动详解(一)之 ...

  7. Fragment详解之三——管理Fragment(1)

    前言:follow your heart,be your own king 相关文章: 1.<Fragment详解之一--概述> 2.<Fragment详解之二--基本使用方法> ...

  8. ViewPager详解(一)——ViewPager的基本使用完整示例

    MainActivity如下: package cn.ww;import java.lang.reflect.Field;import android.app.Activity; import and ...

  9. CORDIC算法详解(五)-统一的 CORDIC 算法形式

    CORDIC算法详解(五)- 统一的 CORDIC 算法形式 文章目录 CORDIC算法详解(五)- 统一的 CORDIC 算法形式 5 统一的 CORDIC 算法形式 相关参考资料及源码   网上有 ...

最新文章

  1. Javascript学习7 - 脚本化浏览器窗口
  2. 修改oracle日志大小,修改Oracle重做日志文件大小
  3. 1257 背包问题 V3——分数规划
  4. 华为电脑c语言总是错误,关于华为机试题求代码!解决方法
  5. linux shell脚本读取配置文件 val=1,shell脚本
  6. IE8 -- 此网页上的问题导致 Internet explore 关闭并重新打开该选项卡
  7. mysql二进制文件下载教程_MySQL二进制版本快速部署
  8. HTML页面中返回顶部的几种实现方式
  9. 数字信号处理——线性相位型(Ⅱ、Ⅳ型)FIR滤波器设计(2)
  10. Agisoft Photoscan/Mateshape 相机参数XML解读
  11. Servlet和Servlet容器概念
  12. mac插网线不能上网_我的Macbook插上网线为什么不能上网? – 手机爱问
  13. 2022年中国大学排行榜出炉~
  14. Java高级技术FastDFS的学习
  15. 「补课」进行时:设计模式(2)——通过一个超级汽车工厂来了解工厂模式
  16. pandas取两个dataframe中不同的行
  17. iPhone/iPad已停用,怎么解锁?
  18. 电脑无法连接wifi得解决方法
  19. rand()函数产生随机数
  20. 野芯CM4_USB3.0x4底板对比树莓派4B的USB3.0速度测试

热门文章

  1. node.js request get 请求怎么拿到返回的数据_NodeJS运维: 从 0 开始 Prometheus + Grafana 业务性能指标监控...
  2. c语言中指针数组赋值字符串,C语言—用结构体指针给数组赋值(结构体指针指向字符串,给字符串赋值)...
  3. 卸料装置弹性零件的计算方法_冲裁模卸料板的设计
  4. tomcat启动占了12g_windows server tomcat服务器部署内存占用高问题
  5. python不同文件函数调用_不同输入文件的多处理调用函数
  6. 电脑光驱不读盘_维修电脑找电脑维修君
  7. matlab中线型和颜色控制
  8. BestCoder-Round#38
  9. 【普及组模拟赛】游戏
  10. 大厅安排(normal)