Android 面试题之 Fragment 使用+实例

举报
帅次 发表于 2021/12/22 23:07:41 2021/12/22
【摘要】 一、为什么要有Fragment? Android运行在各种各样的设备中,有小屏幕的手机,还有大屏幕的平板,电视等。同样的界面在手机上显示可能很好看,在大屏幕的平板上就未必了,手机的界面放在平板上可能会有过分被拉长、控件间距过大等情况。针对屏幕尺寸的差距,Fragment的出现能做到一个App可以同时适应手机和平板。这就是为什么要有...

一、为什么要有Fragment?

Android运行在各种各样的设备中,有小屏幕的手机,还有大屏幕的平板,电视等。同样的界面在手机上显示可能很好看,在大屏幕的平板上就未必了,手机的界面放在平板上可能会有过分被拉长、控件间距过大等情况。针对屏幕尺寸的差距,Fragment的出现能做到一个App可以同时适应手机和平板。这就是为什么要有Fragment的原因。

android.app.Fragment在API 级别 11 中 添加,已在API 级别 28 中弃用

二、Fragment为什么被称为第五大组件

Fragment比Activity更节省内存,其切换模式也更加舒适,使用频率不低于四大组件,且有自己的生命周期,而且必须依附于Activity,不能独立存在。

特点:

  • Fragment依赖于Activity,不能独立存在

  • 一个Activity可以有多个Fragment

  • 一个Fragment可以被多个Activity重用

  • Fragment有自己的生命周期,并能接收输入事件

  • 可以在Activity运行时动态地添加或删除Fragment

三、Activity创建Fragment的方式

静态创建

  1. 定义Fragment的xml布局文件
  2. 自定义Fragment类,继承Fragment类或其子类,同时实现onCreateView()方法,在方法中通过inflater.inflate加载布局文件,接着返回其View
  3. 在需要加载Fragment的Activity对应布局文件中<fragment>的name属性设为全限定类名,即包名.fragment
  4. 最后在Activity调用setContentView()加载布局文件即可

样例:

1.定义Fragment的xml布局文件(新建fragment_home.xml和fragment_mine.xml)

fragment_home.xml:


  
  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:orientation="vertical"
  6. android:background="@color/color_ff0000"
  7. android:gravity="center"
  8. >
  9. <TextView
  10. android:id="@+id/tv_on"
  11. android:layout_width="wrap_content"
  12. android:layout_height="wrap_content"
  13. android:text="首页"
  14. android:onClick="onClick"
  15. android:textColor="@color/white"
  16. android:textSize="@dimen/text_size_20"/>
  17. </LinearLayout>

fragment_mine.xml


  
  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:orientation="vertical"
  6. android:background="@color/color_188FFF"
  7. android:gravity="center"
  8. >
  9. <TextView
  10. android:layout_width="wrap_content"
  11. android:layout_height="wrap_content"
  12. android:text="我的"
  13. android:textColor="@color/white"
  14. android:textSize="@dimen/text_size_20"/>
  15. </LinearLayout>

2.自定义Fragment类,继承Fragment类或其子类,同时实现onCreateView()方法,在方法中通过inflater.inflate加载布局文件,接着返回其View


  
  1. public class HomeFragment extends Fragment {
  2. private TextView tv_on;
  3. @Override
  4. public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  5. View view = inflater.inflate(R.layout.fragment_home, container,false);
  6. MLog.e(this.getClass().getName()+"onCreateView");
  7. tv_on = view.findViewById(R.id.tv_on);
  8. tv_on.setOnClickListener(new View.OnClickListener() {
  9. @Override
  10. public void onClick(View v) {
  11. MLog.e(this.getClass().getName()+"准备关闭");
  12. }
  13. });
  14. return view;
  15. }
  16. }

3.在需要加载Fragment的Activity对应布局文件中<fragment>的name属性设为全限定类名,即包名.fragment.HomeFragment


  
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:id="@+id/ll_bg"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent"
  6. android:background="@color/color_666666"
  7. android:orientation="horizontal">
  8. <fragment
  9. android:id="@+id/fragmen_home"
  10. android:name="com.scc.demo.fragment.HomeFragment"
  11. android:layout_width="0dp"
  12. android:layout_weight="1"
  13. android:layout_height="match_parent" />
  14. <fragment
  15. android:id="@+id/fragmen_mine"
  16. android:name="com.scc.demo.fragment.MineFragment"
  17. android:layout_width="0dp"
  18. android:layout_weight="1"
  19. android:layout_height="match_parent" />
  20. </LinearLayout>

4.最后在Activity调用setContentView()加载布局文件即可


  
  1. public class FragmentActivity extends ActivityBase {
  2. @Override
  3. public void onCreate(Bundle savedInstanceState) {
  4. super.onCreate(savedInstanceState);
  5. MLog.e(this.getClass().getName()+"onCreate");
  6. setContentView(R.layout.activity_fragment);
  7. }
  8. }

动态创建

  1. 获得FragmentManager对象,通过getSupportFragmentManager()

  2. 获得FragmentTransaction对象,通过fm.beginTransaction()

  3. 调用add()方法或者repalce()方法加载Fragment;

  4. 最后调用commit()方法提交事务

样例:

1.activity的布局文件


  
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent">
  5. <FrameLayout
  6. android:id="@+id/fl_frame"
  7. android:layout_width="match_parent"
  8. android:layout_height="match_parent"
  9. android:layout_above="@+id/rg_bottom_tab"/>
  10. <RadioGroup
  11. android:id="@+id/rg_bottom_tab"
  12. android:layout_width="match_parent"
  13. android:layout_height="56dp"
  14. android:layout_alignParentBottom="true"
  15. android:background="#dcdcdc"
  16. android:orientation="horizontal">
  17. <RadioButton
  18. android:id="@+id/rb_home"
  19. android:layout_width="0dp"
  20. android:layout_height="match_parent"
  21. android:layout_weight="1"
  22. android:button="@null"
  23. android:checked="true"
  24. android:gravity="center"
  25. android:text="首页"
  26. android:textSize="@dimen/text_size_18" />
  27. <RadioButton
  28. android:id="@+id/rb_list"
  29. android:layout_width="0dp"
  30. android:layout_height="match_parent"
  31. android:layout_weight="1"
  32. android:button="@null"
  33. android:checked="true"
  34. android:gravity="center"
  35. android:text="列表"
  36. android:textSize="@dimen/text_size_18" />
  37. <RadioButton
  38. android:id="@+id/rb_news"
  39. android:layout_width="0dp"
  40. android:layout_height="match_parent"
  41. android:layout_weight="1"
  42. android:button="@null"
  43. android:checked="true"
  44. android:gravity="center"
  45. android:text="消息"
  46. android:textSize="@dimen/text_size_18" />
  47. <RadioButton
  48. android:id="@+id/rb_mine"
  49. android:layout_width="0dp"
  50. android:layout_height="match_parent"
  51. android:layout_weight="1"
  52. android:button="@null"
  53. android:gravity="center"
  54. android:text="我的"
  55. android:textSize="@dimen/text_size_18" />
  56. </RadioGroup>
  57. </RelativeLayout>

2.自定义Fragment


  
  1. public class HomeFragment extends Fragment {
  2. private TextView tv_on;
  3. public static HomeFragment newInstance(String param){
  4. HomeFragment fragment = new HomeFragment();
  5. Bundle args = new Bundle();
  6. args.putString("param", param);
  7. fragment.setArguments(args);
  8. return fragment;
  9. }
  10. public HomeFragment(){}
  11. @Override
  12. public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  13. View view = inflater.inflate(R.layout.fragment_home, container,false);
  14. MLog.e(this.getClass().getName()+"onCreateView");
  15. Bundle bundle = getArguments();
  16. String param = bundle.getString("param");
  17. tv_on = view.findViewById(R.id.tv_on);
  18. tv_on.setText("首页"+param);
  19. tv_on.setOnClickListener(new View.OnClickListener() {
  20. @Override
  21. public void onClick(View v) {
  22. MLog.e(this.getClass().getName()+"准备关闭");
  23. }
  24. });
  25. return view;
  26. }
  27. }

3.获得FragmentManager对象,通过getFragmentManager()


  
  1. public class FragmentActivity extends ActivityBase {
  2. private RadioGroup rg_bottom_tab;
  3. private SparseArray<Fragment> mFragmentList;
  4. @Override
  5. public void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_fragment);
  8. MLog.e(this.getClass().getName()+"onCreate");
  9. initView();
  10. }
  11. public void initView(){
  12. rg_bottom_tab = findViewById(R.id.rg_bottom_tab);
  13. mFragmentList = new SparseArray<>();
  14. mFragmentList.append(R.id.rb_home, HomeFragment.newInstance("我最帅"));
  15. mFragmentList.append(R.id.rb_list, ListFragment.newInstance("我最美"));
  16. mFragmentList.append(R.id.rb_news, NewsFragment.newInstance("我最新"));
  17. mFragmentList.append(R.id.rb_mine, MineFragment.newInstance("这里是我的"));
  18. rg_bottom_tab.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
  19. @Override
  20. public void onCheckedChanged(RadioGroup group, int checkedId) {
  21. // 具体的fragment切换逻辑可以根据应用调整,例如使用show()/hide()
  22. FragmentManager fragmentManager = getFragmentManager();
  23. FragmentTransaction transaction = fragmentManager.beginTransaction();
  24. transaction.replace(R.id.fl_frame, mFragmentList.
  25. get(checkedId));
  26. transaction.commit();
  27. }
  28. });
  29. // 默认显示第一个
  30. FragmentManager fragmentManager = getFragmentManager();
  31. FragmentTransaction transaction = fragmentManager.beginTransaction();
  32. transaction.add(R.id.fl_frame, mFragmentList.get(R.id.rb_home)).commit();
  33. }
  34. }

四、FragmentPageAdapter和FragmentPageStateAdapter的区别

FragmentPageAdapter在每次切换页面的的时候,是将Fragment进行分离,适合页面较少的Fragment使用以保存一些内存,对系统内存不会多大影响。

FragmentPageStateAdapter在每次切换页面的时候,是将Fragment进行回收,适合页面较多的Fragment使用,这样就不会消耗更多的内存。

五、Fragment生命周期

被调用以使Fragment恢复到恢复状态(与用户交互)的核心生命周期方法系列是:

onAttach(Activity) 一旦Fragment与其活动相关联,就会调用。

onCreate(Bundle) 调用以进行Fragment的初始创建。

onCreateView(LayoutInflater, ViewGroup, Bundle) 创建并返回与Fragment关联的视图层次结构。

onActivityCreated(Bundle)告诉Fragment它的活动已经完成了它自己的Activity.onCreate().

onViewStateRestored(Bundle) 告诉Fragment其视图层次结构的所有保存状态都已恢复。

onStart() 使Fragment对用户可见(基于正在启动的包含活动)。

onResume() 使Fragment开始与用户交互(基于其包含的活动正在恢复)。

由于不再使用Fragment,它会经历一系列反向回调:

onPause() Fragment不再与用户交互,因为其活动被暂停或片段操作正在活动中修改它。

onStop() Fragment不再对用户可见,因为其活动正在停止或片段操作正在活动中修改它。

onDestroyView() 允许Fragment清理与其视图关联的资源。

onDestroy() 调用对Fragment状态进行最终清理。

onDetach() 在Fragment不再与其活动相关联之前立即调用。

六、Fragment的通信

Fragment调用Activity中的方法:getActivity返回当前与此Fragment关联的 Activity。

Activity调用Fragment中的方法:接口回调

Fragment调用Fragment中的方法:通过Activity中的FragmentManager

七、Fragment的状态保存

Fragment负责管理与 Fragment功能相关的少量动态状态。你可以使用 Fragment.onSaveInstanceState(Bundle).类似 Activity.onSaveInstanceState(Bundle),你的包将数据通过配置的变化和处理死亡和娱乐保留,并在你的碎片的可用 onCreate(Bundle),onCreateView(LayoutInflater,ViewGroup, Bundle)和 onViewCreated(View,Bundle) 方法。

如上图所示,onStop()回调的顺序 和状态的保存因onSaveInstanceState()API 级别而异。对于 API 28 之前的所有 API 级别,onSaveInstanceState()在onStop(). 对于 API 级别 28 及更高级别,调用顺序相反。

八、Fragment的replace、add、remove方法

replace:替代Fragment的栈顶页面

add:添加Fragment到栈顶页面

remove:移除Fragment栈顶页面

九、Fragment的优势

Fragment可以使你能够将activity分离成多个可重用的组件,每个都有它自己的生命周期和UI。

Fragment可以轻松得创建动态灵活的UI设计,可以适应于不同的屏幕尺寸。从手机到平板电脑。

Fragment是一个独立的模块,紧紧地与activity绑定在一起。可以运行中动态地移除、加入、交换等。

Fragment提供一个新的方式让你在不同的安卓设备上统一你的UI。

Fragment 解决Activity间的切换不流畅,轻量切换。

Fragment 替代TabActivity做导航,性能更好。

Fragment 在4.2.版本中新增嵌套fragment使用方法,能够生成更好的界面效果。

Fragment做局部内容更新更方便,原来为了到达这一点要把多个布局放到一个activity里面,现在可以用多Fragment来代替,只有在需要的时候才加载Fragment,提高性能。

可以从startActivityForResult中接收到返回结果,但是View不能。

总结下来:

模块化(Modularity):我们不必把所有代码全部写在Activity中,而是把代码写在各自的Fragment中。

可重用(Reusability):多个Activity可以重用一个Fragment。

可适配(Adaptability):根据硬件的屏幕尺寸、屏幕方向,能够方便地实现不同的布局,这样用户体验更好。

十、androidx包访问Fragment

在Activity中访问

每个FragmentActivity及其子类(如AppCompatActivity)都可以通过getSupportFragmentManager()方法访问FragmentManager。

在Fragment中访问

Fragment也能够托管一个或多个子Fragment。在Fragment内,你可以通过getChildFragmentManager()获取对管理Fragment子级的FragmentManager的引用。如果你需要访问其宿主FragmentManager,可以使用getParentFragmentManager()。

文章来源: shuaici.blog.csdn.net,作者:帅次,版权归原作者所有,如需转载,请联系作者。

原文链接:shuaici.blog.csdn.net/article/details/117930956

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。