Android中Fragment相互切换间不被回收的实现方法
时间:2022-07-27
本文章向大家介绍Android中Fragment相互切换间不被回收的实现方法,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
前言
Android运行在各种各样的设备中,有小屏幕的手机,超大屏的平板甚至电视。针对屏幕尺寸的差距,很多情况下,都是先针对手机开发一套App,然后拷贝一份,修改布局以适应平板神马超级大屏的。难道无法做到一个App可以同时适应手机和平板么,当然了,必须有啊。Fragment的出现就是为了解决这样的问题。
如今市面上的应用基本上都是单Activity+多Fragment实现的了,而这类APP都有在相互切换时不被回收,即切换回原来的Fragment时还是原先的状态,这就是这里要实现的了。
这里使用Fragment的add()
、show()
、hide()
实现,即显示和隐藏,这样原来的Fragment就不会被销毁了。
二话不说,贴代码,代码是最好的老师。
示例代码(注释还算详细了)
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private ImageView ibOne;
private ImageView ibTwo;
private ImageView ibThree;
private FragmentManager mFm;
private ArrayList<Fragment mFragmentList = new ArrayList<Fragment ();
private String[] mFragmentTagList = {"OneFragment", "TwoFragment", "ThreeFragment"};
private Fragment mCurrentFragmen = null; // 记录当前显示的Fragment
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
initData();
}
private void initData() {
OneFragment oneFragment = new OneFragment();
TwoFragment twoFragment = new TwoFragment();
ThreeFragment threeFragment = new ThreeFragment();
mFragmentList.add(0, oneFragment);
mFragmentList.add(1, twoFragment);
mFragmentList.add(2, threeFragment);
mCurrentFragmen = mFragmentList.get(0);
// 初始化首次进入时的Fragment
mFm = getFragmentManager();
FragmentTransaction transaction = mFm.beginTransaction();
transaction.add(R.id.fl_show, mCurrentFragmen, mFragmentTagList[0]);
transaction.commitAllowingStateLoss();
}
// findViewById
private void initView() {
ibOne = (ImageView)findViewById(R.id.ib_one);
ibTwo = (ImageView)findViewById(R.id.ib_two);
ibThree = (ImageView)findViewById(R.id.ib_three);
ibOne.setOnClickListener(this);
ibTwo.setOnClickListener(this);
ibThree.setOnClickListener(this);
}
@Override
public void onClick(View view) {
switch (view.getId()){
case R.id.ib_one:
switchFragment(mFragmentList.get(0), mFragmentTagList[0]);
break;
case R.id.ib_two:
switchFragment(mFragmentList.get(1), mFragmentTagList[1]);
break;
case R.id.ib_three:
switchFragment(mFragmentList.get(2), mFragmentTagList[2]);
break;
}
}
// 转换Fragment
void switchFragment(Fragment to, String tag){
if(mCurrentFragmen != to){
FragmentTransaction transaction = mFm.beginTransaction();
if(!to.isAdded()){
// 没有添加过:
// 隐藏当前的,添加新的,显示新的
transaction.hide(mCurrentFragmen).add(R.id.fl_show, to, tag).show(to);
}else{
// 隐藏当前的,显示新的
transaction.hide(mCurrentFragmen).show(to);
}
mCurrentFragmen = to;
transaction.commitAllowingStateLoss();
}
}
// 当activity非正常销毁时被调用
@Override
public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) {
super.onSaveInstanceState(outState, outPersistentState);
// 重置Fragment,防止当内存不足时导致Fragment重叠
updateFragment(outState);
}
// 重置Fragment
private void updateFragment(Bundle outState) {
mFm = getFragmentManager();
if(outState == null){
FragmentTransaction transaction = mFm.beginTransaction();
OneFragment oneFragment = new OneFragment();
mCurrentFragmen = oneFragment;
transaction.add(R.id.fl_show, oneFragment, mFragmentTagList[0]).commitAllowingStateLoss();
}else{
// 通过tag找到fragment并重置
OneFragment oneFragment = (OneFragment) mFm.findFragmentByTag(mFragmentTagList[0]);
TwoFragment twoFragment = (TwoFragment) mFm.findFragmentByTag(mFragmentTagList[1]);
ThreeFragment threeFragment = (ThreeFragment) mFm.findFragmentByTag(mFragmentTagList[2]);
mFm.beginTransaction().show(oneFragment).hide(twoFragment).hide(threeFragment);
}
}
}
我以前对于这种需求是在一个Activity中使用RelativeLayout,在其中加入多个布局(类似Fragment),当点击下方Tab时设置布局的visibility的,思想是一样的,但这样实现起来很是丑陋,所以不建议使用。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对ZaLou.Cn的支持
- Dora.Interception, 为.NET Core度身打造的AOP框架:不一样的Interceptor定义方式
- Dora.Interception,为.NET Core度身打造的AOP框架:全新的版本
- ASP.NET Core的路由[4]:来认识一下实现路由的RouterMiddleware中间件
- 浅谈 Java 并发编程中的若干核心技术
- ASP.NET Core的路由[3]:Router的创建者——RouteBuilder
- ASP.NET Core的路由[2]:路由系统的核心对象——Router
- ASP.NET Core的路由[1]:注册URL模式与HttpHandler的映射关系
- 学习ASP.NET Core, 怎能不了解请求处理管道[6]: 管道是如何随着WebHost的开启被构建出来的?
- 学习ASP.NET Core, 怎能不了解请求处理管道[5]: 中间件注册可以除了可以使用Startup之外,还可以选择StartupFilter
- 学习ASP.NET Core, 怎能不了解请求处理管道[4]: 应用的入口——Startup
- 学习ASP.NET Core, 怎能不了解请求处理管道[3]: 自定义一个服务器感受一下管道是如何监听、接收和响应请求的
- .NET Core多平台开发体验[4]: Docker
- .NET Core多平台开发体验[3]: Linux (Windows Linux子系统)
- .NET Core多平台开发体验[2]: Mac OS X
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释