在实际项目的开发过程Fragment的情况越来越多。大家肯定须要遇到过Fragment被销毁重建的情况。

结合自己在项目开发的一点总结和学习开源项目的代码。继续分享自己对Fragment的一点总结。


1.Fragment保存销毁前状态究竟保存什么?

我们知道Fragment的实例会在多种情况下被系统销毁回收掉。当我们的Fragment又一次回到屏幕前,我们想要的还是销毁前的状态。因此。我们在Fragment被销毁掉的时候,我们须要保存Fragment的状态。

以下回到我们的问题,Fragment保存销毁前状态究竟保存什么?TextView的文字,还是WebView的载入内容?好像问题有点复杂。 Android的系统设计者已经为我们设计了一套保存机制,我们根本不用去考虑这些问题。 我们看看TextView的源代码的一段代码:

@Override
public Parcelable onSaveInstanceState() {
Parcelable superState = super.onSaveInstanceState(); } @Override
public void onRestoreInstanceState(Parcelable state) {
if (!(state instanceof SavedState)) {
super.onRestoreInstanceState(state);
return;
} SavedState ss = (SavedState) state;
super.onRestoreInstanceState(ss.getSuperState());
}

看上面的代码相信从字面意思大家也能明确。上面的代码保存TextView的状态和控件恢复时候状态恢复。我们去查看ListView的源代码。我们基类AbsListView的源代码会也看到类似onSaveInstanceState、onRestoreInstanceState方法。对ListView控件,这里有一点须要我们注意,当ListView销毁恢复的时候。我们不能恢复ListView到绑定数据的状态。由于ListView的实现是基于适配器模式的设计实现,ListView 仅仅不关心数据负责展示数据。不关心数据源。

如今我们知道Fragment状态的恢复。由于谷歌的攻城狮的强大机制的设计,我们不用去考虑控件本身状态的恢复(ListView等控件数据的恢复以下会说道)。那哪些是我们要考虑的呢?能够总结下例如以下:

  • 我们要考虑第三方控件的实现。在恢复控件状态的时候是否考虑全面;
  • 对 Fragment的成员变量的状态保存(ScrollView当前滑动到的位置、用户操作标记的图片等)。

第一种情况不是我们这次讨论的重点,我们主要来讨论另外一种情况的状态保存。结合自己的项目里的经验,我们有两种保存方式:

第一种实现方式:

我们来结合对ListView的数据状态恢复来说明对Fragment的状态恢复的标准实现。

public class XXXFragment extends Fragment {

    private static final String VIEW_POSITION = "view_postion";
private ListView listView; @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View rootView = (ViewGroup) inflater.inflate(
R.layout.order_list_fragment, container, false);
initView(rootView, inflater);
if (savedInstanceState != null) { // biaoshi
mCurrentPage = savedInstanceState.getInt(VIEW_POSITION);
// 再次又一次请求数据(从缓存)。listView绑定数据,又一次设置上次浏览的位置mCurrentPage
} else {
// 首次请求数据,listView绑定数据
}
return rootView;
} /*
* 保存当前Frament状态
*/
@Override
public void onSaveInstanceState(Bundle outState) {
outState.putInt(VIEW_POSITION, mCurrentPage);
super.onSaveInstanceState(outState);
}
}

第一种实现方式:

通过学习chrisbanes大神的photup项目,我们把Fragment的状态用Controller 类对象记录。该Controller在 Application的onCreate完毕初始化,我们能够理解Fragment的状态我们用全局变量在记录。由于photup项目对Fragment的状态信息的管理,我们能够不用操心Fragment被销毁时候状态保存,我们在每次调用Fragment的时候,我们又一次实例化一个新的Fragment再恢复销毁前的状态。详细项目实现我这里就不给出来了。參看chrisbanes的photup项目实现。

针对chrisbanes的实现,我个人认为有些看法:

1.我认为这样的实现方式(Fragment + Controller)的长处在于,方便同一个Activity下多个Fragment的沟通实现,作为一个小项目的实现还是非常灵活方便。

假设开发一个业务繁杂的项目,这样的方式实现会造成过多全局Congtroller对象。

2.我认为。这样Fragment + Controller的方式实现Service的时候能够借鉴,一般项目的Service都不会太多。并且(Service + Controller)的方式也方面Service和其它组件沟通。

对Fragment销毁和重建的管理方式

在学习wordpress-androi的源代码实现的时候,学习到wordpress通过ViewPager来实现对Activity下的Fragment的管理方式最大简化我们对Fragment销毁和重建的管理。

(ps:Wordpress-Android源代码值得学习的地方远不止这一点,假设大家感兴趣,去看看wordpress-android的实现。一定会有不少收获)。

对Fragment的管理一些注意的地方,在我前一篇博客 Fragment的实际开发的一些总结 里说道一些关于Fragment被系统销毁重建管理的一些问题。我们通过前面模仿模仿QQ首页的样例通过wordpress-android的方式来实现一次。

我们先重写ViewPager一些方法:

public class HViewPager extends ViewPager {
private boolean mPagingEnabled = true; public HViewPager(Context context) {
super(context);
} public HViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
} @Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (mPagingEnabled) {
try {
return super.onInterceptTouchEvent(ev);
} catch (IllegalArgumentException e) { }
}
return false;
} @Override
public boolean onTouchEvent(MotionEvent ev) {
if (mPagingEnabled) {
try {
return super.onTouchEvent(ev);
} catch (IllegalArgumentException e) { }
}
return false;
} /*
* 设置是否关闭ViewPager的滑动效果(我们能够实现ViewPager的Fragment间切换如Activity切换的感觉)
*/
public void setPagingEnabled(boolean pagingEnabled) {
mPagingEnabled = pagingEnabled;
}
}

然后我们用HViewPager相应的FragmentPagerAdapter子类SwitchPagerAdapter负责管理Fragment,简单实现例如以下:

public class SwitchPagerAdapter extends FragmentPagerAdapter {

    private static final int NUM_TABS = 2;
private static final int TAB_MESSAGE = 0;
private static final int TAB_CALL = 1; public SwitchPagerAdapter(FragmentManager fm) {
super(fm);
} @Override
public int getCount() {
return NUM_TABS;
} @Override
public Fragment getItem(int position) {
switch (position) {
case TAB_MESSAGE:
return new MessageFragment();
case TAB_CALL:
return new CallFragment();
default:
return null;
}
}
}

我们再来看首页的切换效果如今例如以下所看到的:

public class SwitchActivity extends FragmentActivity {

    private Button btn_message, btn_call;
private HViewPager viewpager; public static final int MESSAGE_FRAGMENT_TYPE = 1;
public static final int CALL_FRAGMENT_TYPE = 2;
public int currentFragmentType = -1; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_switch); btn_message = (Button) findViewById(R.id.btn_message);
btn_call = (Button) findViewById(R.id.btn_call);
viewpager = (HViewPager) findViewById(R.id.viewpager);
SwitchPagerAdapter adapter = new SwitchPagerAdapter(
getSupportFragmentManager());
viewpager.setPagingEnabled(false);
viewpager.setAdapter(adapter);
btn_message.setOnClickListener(onClicker);
btn_call.setOnClickListener(onClicker); if (savedInstanceState != null) {
int index = savedInstanceState.getInt("currentFragmentType");
if (index > 0)
switchPager(1);
else
switchPager(0);
} else {
switchPager(0);
} } private void switchPager(int index) {
viewpager.setCurrentItem(index);
currentFragmentType = index;
} /*
* 保存当前展示那个页面
*/
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putInt("lastFragmentTag", currentFragmentType);
} private OnClickListener onClicker = new OnClickListener() {
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_message:
btn_message.setTextColor(Color.parseColor("#df3031"));
btn_call.setTextColor(Color.WHITE);
btn_message
.setBackgroundResource(R.drawable.baike_btn_pink_left_f_96);
btn_call.setBackgroundResource(R.drawable.baike_btn_trans_right_f_96);
switchPager(0); break;
case R.id.btn_call: btn_message.setTextColor(Color.WHITE);
btn_call.setTextColor(Color.parseColor("#df3031"));
btn_message
.setBackgroundResource(R.drawable.baike_btn_trans_left_f_96);
btn_call.setBackgroundResource(R.drawable.baike_btn_pink_right_f_96);
switchPager(1); break; }
}
};
}

以上是通过ViewPager来达到管理Fragment 销毁和重建的实现。在WordPress-android源代码实现里关于Fragment的管理原理上差点儿相同这样,当然WordPress-android的实如今代码编写上能够学习的地方远不止如此。除了代码编写上。WordPress-android界面效果利用Fragment达到了非常多非常不错的使用效果。以下贴出WordPress-Android的博客编辑器编辑界面仅仅用一个Activity+多个Fragment的实现效果。实现还是相当给力的。





ps:第一次用 markdown编辑器。真心还有点不习惯,格式学习中。

转载请注明出处:http://blog.csdn.net/johnnyz1234/

Fragment的实际开发中总结(二)的更多相关文章

  1. 微信小程序开发中的二三事之网易云信IMSDK DEMO

    本文由作者邹永胜授权网易云社区发布. 简介 为了更好的展示我们即时通讯SDK强悍的能力,网易云信IM SDK微信小程序DEMO的开发就提上了日程.用产品的话说就是: 云信 IM 小程序 SDK 的能力 ...

  2. Java 数据类型在实际开发中应用二枚举

    在实际编程中,往往存在着这样的"数据集",它们的数值在程序中是稳定的,而且"数据集"中的元素是有限的.在JDK1.5之前,人们用接口来描述这一种数据类型. 1. ...

  3. java和h5 canvas德州扑克开发中(二)

    德州扑克网页源码在github上分享 https://github.com/lxr1907/pokers 感兴趣的可以上去看下. 1.通讯使用websocket,主要在message.js中. 2.用 ...

  4. iOS开发-定制多样式二维码

    iOS开发-定制多样式二维码   二维码/条形码是按照某种特定的几何图形按一定规律在平台(一维/二维方向上)分布的黑白相间的图形纪录符号信息.使用若干个与二进制对应的几何形体来表示文字数值信息. 最常 ...

  5. android开发中fragment获取context

    在用到fragment时无法使用.this来指定当前context内容,android开发中fragment获取context,可以使用getActivity().getApplicationCont ...

  6. iOS开发中的4种数据持久化方式【二、数据库 SQLite3、Core Data 的运用】

                   在上文,我们介绍了ios开发中的其中2种数据持久化方式:属性列表.归档解档.本节将继续介绍另外2种iOS持久化数据的方法:数据库 SQLite3.Core Data 的运 ...

  7. 二、Cocos2dx概念介绍(游戏开发中不同的坐标系,cocos2dx锚点)

    注:ccp是cocos2dx中的一个宏定义,#define ccp(__X__,__Y__)CCPointMake((float)__X__, (float)__Y__),在此文章中表示坐标信息 1. ...

  8. SQL开发中容易忽视的一些小地方(二)

    原文:SQL开发中容易忽视的一些小地方(二) 目的:继上一篇:SQL开发中容易忽视的一些小地方(一) 总结SQL中的null用法后,本文我将说说表联接查询. 为了说明问题,我创建了两个表,分别是学生信 ...

  9. 设计模式笔记之二:Android开发中的MVP架构(转)

    写在前面,本博客来源于公众号文章:http://mp.weixin.qq.com/s?__biz=MzA3MDMyMjkzNg==&mid=402435540&idx=1&sn ...

随机推荐

  1. 虚拟机创建后该如何获取IP地址并访问互联网实用教程

    之前在做项目的时候主机IP地址.网关.DNS.子网掩码等都是公司或者对方直接给提供的,但是如果我们自己想搭建一台虚拟机或者一台集群的话,手头又没有IP地址,该肿么办呢?   白慌,这里介绍一个小技巧, ...

  2. javascript 继承之拷贝,原型,类式

    // 拷贝继承,在子类内调用父类并修正this指向,再通过for in 拷贝父类的方法实现继承,具体实现如下代码 : function Tab(){//父类构造函数 this.name='aaa'; ...

  3. axios简单封装

    写在最前面 新手前端刚刚接触vue,感觉真的好用.项目中需要使用axios,然后学习了一下.借鉴网上一些大佬的经验,现在分享一下axios的简单封装,如果有什么错误的地方,请大家指出. axios安装 ...

  4. Mysql学习总结(28)——MySQL建表规范与常见问题

    一. 表设计 库名.表名.字段名必须使用小写字母,"_"分割. 库名.表名.字段名必须不超过12个字符. 库名.表名.字段名见名知意,建议使用名词而不是动词. 建议使用InnoDB ...

  5. 设计模式-策略模式(Go语言描写叙述)

    好久没有更新博客了.近期也是在忙着充电,今天这篇博客開始,我们来了解一下设计模式. 设计模式 那什么是设计模式呢?首先来看看我从百科上copy下来的概念吧. 设计模式/软件设计模式(Design pa ...

  6. linux线程间同步(1)读写锁

    读写锁比mutex有更高的适用性,能够多个线程同一时候占用读模式的读写锁.可是仅仅能一个线程占用写模式的读写锁. 1. 当读写锁是写加锁状态时,在这个锁被解锁之前,全部试图对这个锁加锁的线程都会被堵塞 ...

  7. 基于SIP和RTP协议的开源VOIP之QuteCom简单介绍

    **************************************************************************************************** ...

  8. What is the difference between SET and SELECT when assigning values to variables, in T-SQL?

    http://vyaskn.tripod.com/differences_between_set_and_select.htm https://stackoverflow.com/questions/ ...

  9. Kali linux 2016.2(Rolling)中metasploit的端口扫描

    目前常见的端口扫描技术一般有如下几类: TCP  Connect.TCP SYN.TCP ACK.TCP FIN. Metasploit中的端口扫描器 Metasploit的辅助模块中提供了几款实用的 ...

  10. HDU 3342 Legal or Not(判断环)

    Problem Description ACM-DIY is a large QQ group where many excellent acmers get together. It is so h ...