现在越来越多的应用会使用viewpager+fragment显示自己的内容页,fragment和activity有很多共同点,如下图就是fragment的生命周期

但是fragment和activity不同的是当调用本身的onResume和onPause方法的时候可能并不是当前的fragment在显示,例如当加载下面这张图时,当我打开MainActivity时显示的是第一个fragment 但此时调用的方法步骤如下:

);

运行时打印的log和上面完全一致,即就算你设置只加载一个fragment还是会加载第二个fragment,原因是setOffscreenPageLimit中的源码时这样写的

[java] view plain copy
  1. public void setOffscreenPageLimit(int limit) {
  2. if (limit < DEFAULT_OFFSCREEN_PAGES) {
  3. Log.w(TAG, "Requested offscreen page limit " + limit + " too small; defaulting to " +
  4. DEFAULT_OFFSCREEN_PAGES);
  5. limit = DEFAULT_OFFSCREEN_PAGES;
  6. }
  7. if (limit != mOffscreenPageLimit) {
  8. mOffscreenPageLimit = limit;
  9. populate();
  10. }
  11. }

这个 DEFAULT_OFFSCREEN_PAGES 定义如下 private static final intDEFAULT_OFFSCREEN_PAGES= 1;

就是如果你设置为0 也没用。!!!

为了解决判断是否fragment当前显示问题 可以在fragment重写 setUserVisibleHint(boolean isVisibleToUser)

在fragment添加log日志

[java] view plain copy
  1. public class Fragment1 extends Fragment {
  2. private static final String TAG = "Fragment1";
  3. @Override
  4. public void onAttach(Activity activity) {
  5. super.onAttach(activity);
  6. L.v(TAG, "onAttach");
  7. }
  8. @Override
  9. public void setUserVisibleHint(boolean isVisibleToUser) {
  10. L.v(TAG, "setUserVisibleHint " + isVisibleToUser);
  11. }
  12. @Override
  13. public void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. L.v(TAG, "onCreate");
  16. }
  17. @Override
  18. public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  19. L.v(TAG, "onCreateView");
  20. View view = inflater.inflate(R.layout.fragment_layout,null);
  21. TextView tv = (TextView) view.findViewById(R.id.tv1);
  22. tv.setText(TAG);
  23. return view;
  24. }
  25. @Override
  26. public void onActivityCreated(@Nullable Bundle savedInstanceState) {
  27. super.onActivityCreated(savedInstanceState);
  28. L.v(TAG,"onActivityCreated");
  29. }
  30. @Override
  31. public void onResume() {
  32. super.onResume();
  33. L.v(TAG, "onResume()");
  34. }
  35. @Override
  36. public void onPause() {
  37. super.onPause();
  38. L.v(TAG,"onPause");
  39. }
  40. @Override
  41. public void onStop() {
  42. super.onStop();
  43. L.v(TAG,"onStop");
  44. }

启动activity打印日志如下:

08-11 11:33:36.156    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint false
08-11 11:33:36.156    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint false
08-11 11:33:36.157    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint true
08-11 11:33:36.158    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onAttach
08-11 11:33:36.158    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onCreate
08-11 11:33:36.159    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onCreateView
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onActivityCreated
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onResume()
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onAttach
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onCreate
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onCreateView
08-11 11:33:36.161    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onActivityCreated
08-11 11:33:36.161    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onResume()

当切换到第二个fragment时打印日志:

08-11 11:33:54.084    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ setUserVisibleHint false
08-11 11:33:54.084    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint false
08-11 11:33:54.084    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint true
08-11 11:33:54.084    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ onAttach
08-11 11:33:54.085    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ onCreate
08-11 11:33:54.085    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ onCreateView
08-11 11:33:54.085    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ onActivityCreated
08-11 11:33:54.085    7162-7162/com.example.yinsgo.myui V/Fragment3﹕ onResume()

可见当fragment显示时回调用方法 setUserVisibleHint中的isVisibleToUser = true 当fragment被切换隐藏时回 isVisibleToUser = false;

所以当我们要统计是否用户看到一个fragment时可以执行一下代码

[java] view plain copy
  1. @Override
  2. public void setUserVisibleHint(boolean isVisibleToUser) {
  3. L.v(TAG, "setUserVisibleHint " + isVisibleToUser);
  4. if (isVisibleToUser) {
  5. //统计代码 或者 fragment显示操作
  6. } else {
  7. }
  8. }

同时根据这个方法还可以进行数据的延迟加载,后面再写。

今天又看了一下发现单纯的执行上面的代码是有问题的,因为如果我们在else中认为用户是离开界面其实是不对的,因为根据启动第一个Fragment的log日志

08-11 11:33:36.156    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint false
08-11 11:33:36.156    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint false
08-11 11:33:36.157    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint true
08-11 11:33:36.158    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onAttach
08-11 11:33:36.158    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onCreate
08-11 11:33:36.159    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onCreateView
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onActivityCreated
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment1﹕ onResume()
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onAttach
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onCreate
08-11 11:33:36.160    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onCreateView
08-11 11:33:36.161    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onActivityCreated
08-11 11:33:36.161    7162-7162/com.example.yinsgo.myui V/Fragment2﹕ onResume()

第一次setUserVisibleHint 方法 isVisibleToUser 是false 但其实这个时候只是还未初始化,并不是用户已经浏览界面准备离开,于是这里我们需要一个辅助标记变量具体代码如下:

[java] view plain copy
  1. /**
  2. * 判断是否是初始化Fragment
  3. */
  4. private boolean hasStarted = false;
  5. @Override
  6. public void setUserVisibleHint(boolean isVisibleToUser) {
  7. super.setUserVisibleHint(isVisibleToUser);
  8. L.v(TAG, "setUserVisibleHint " + isVisibleToUser);
  9. if (isVisibleToUser) {
  10. hasStarted = true;
  11. L.v(TAG,"开始界面");
  12. } else {
  13. if (hasStarted) {
  14. hasStarted = false;
  15. L.v(TAG,"结束界面");
  16. }
  17. }
  18. }

当我们启动MainActivity时 Log打印如下:

08-13 17:55:45.850  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint false
08-13 17:55:45.850  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint false
08-13 17:55:45.850  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onAttach
08-13 17:55:45.850  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onCreate
08-13 17:55:45.850  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint true
08-13 17:55:45.860  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ 开始界面
08-13 17:55:45.860  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onCreateView
08-13 17:55:45.870  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onActivityCreated
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onResume()
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ onAttach
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ onCreate
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ onCreateView
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ onActivityCreated
08-13 17:55:45.880  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ onResume()

切换到第二个fragment时,此时离开第一个fagment  log打印如下

08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ setUserVisibleHint false
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ setUserVisibleHint false
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ 结束界面
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint true
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ 开始界面
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ onAttach
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ onCreate
08-13 17:57:04.310  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ onCreateView
08-13 17:57:04.320  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ onActivityCreated
08-13 17:57:04.320  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ onResume()

切换到第三个fragment时,此时离开第二个fragment log打印如下:

08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ setUserVisibleHint false
08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment2﹕ 结束界面
08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ setUserVisibleHint true
08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment3﹕ 开始界面
08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onPause
08-13 17:58:15.040  21467-21467/com.example.yinsgo.myui V/Fragment1﹕ onStop

可见这样就可以准确统计用户是否离开或者开始浏览界面了。根据这些开始和离开可以统计用户停留界面的时间等数据。

最新文章

  1. 将excel数据读入matlab
  2. Android 数据存储五种方式
  3. 理解OAuth2.0
  4. AngularJS 开发中常犯的10个错误
  5. ajax小技巧,防止多次点击发送多个请求
  6. oracle创建索引后sqlldr导入错误
  7. Mysql按时间段分组查询来统计会员的个数
  8. rtp h264注意点(FU-A分包方式说明)
  9. MySQL查询in操作 查询结果按in集合顺序显示(转)
  10. [jstips]undefined和null的区别
  11. 填写信息的文章区域text_area
  12. iOS的横屏(Landscape)与竖屏(Portrait)InterfaceOrientation
  13. poj-1503-java大数相加
  14. Linux内存管理 (19)总结内存管理数据结构和API
  15. Visual Studio 2012编译的程序无法在XP下运行的解决办法【转】
  16. 关于python的一些想法
  17. python3+redis问题求解
  18. Linux C Socket TCP编程介绍及实例
  19. flash builder 4.7 打开闪退解决办法
  20. 【转】 Android常用实例—Alert Dialog的使用

热门文章

  1. Linux内存管理学习2 —— head.S中的段页表的建立
  2. MAC系统压缩文件传到WINDOWS下出现乱码
  3. Ora-01536:超出了表空间users的空间限量
  4. Raspbian 中国软件源
  5. MySql错误处理(三)- 错误处理的例子
  6. xheditor-文件上传-java-支持html5-application/octet-stream
  7. System.loadLibrary()的使用方法汇总
  8. Eclipse 保存文件时自动格式化代码
  9. sublime text的扩展插件
  10. ExtJS 4.2 教程-05:客户端代理(proxy)