• Android Fragment


    基本概念

    Fragment是Android3.0后引入的一个新的API,他出现的初衷是为了适应大屏幕的平板电脑, 普通手机开发也会加入这个Fragment, 可以把他看成一个小型的Activity,又称Activity片段!

    如果一个很大的界面,就一个布局,写起界面会很麻烦,而且如果组件多的话是管理起来也很麻烦!而使用Fragment 我们可以把屏幕划分成几块,然后进行分组,进行一个模块化的管理!从而可以更加方便的在 运行过程中动态地更新Activity的用户界面!

    另外Fragment并不能单独使用,他需要嵌套在Activity 中使用,尽管他拥有自己的生命周期,但是还是会受到宿主Activity的生命周期的影响,比如Activity 被destory销毁了,他也会跟着销毁!

    下图是文档中给出的一个Fragment分别对应手机与平板间不同情况的处理图:

    PS: 简单的新闻浏览页面,使用两个Fragment分别显示新闻列表与新闻内容

    Fragment的生命周期图

    使用Fragment的一些要点

    Fragment需要嵌套在Activity中使用,当然也可以嵌套到另外一个Fragment中,但这个被嵌套 的Fragment也是需要嵌套在Activity中的,间接地说,Fragment还是需要嵌套在Activity中!! 受寄主Activity的生命周期影响,当然他也有自己的生命周期!另外不建议在Fragment里面 嵌套Fragment因为嵌套在里面的Fragment生命周期不可控!!!

    官方文档说创建Fragment时至少需要实现三个方法:onCreate( ),onCreateView( ),OnPause( ); 不过貌似只写一个onCreateView也是可以的

    Fragment的生命周期和Activity有点类似:三种状态:
    Resumed:在允许中的Fragment可见
    Paused:所在Activity可见,但是得不到焦点
    Stoped: ①调用addToBackStack(),Fragment被添加到Bcak栈 ②该Activity转向后台,或者该Fragment被替换/删除

    ps:停止状态的fragment仍然活着(所有状态和成员信息被系统保持着),然而,它对用户 不再可见,并且如果activity被干掉,他也会被干掉.

    Fragment的几个子类

    ps:很多时候我们都是直接重写Fragment,inflate加载布局完成相应业务了,子类用的不多,等需要的 时候在深入研究!

    • 对话框:DialogFragment
    • 列表:ListFragment
    • 选项设置:PreferenceFragment
    • WebView界面:WebViewFragment

    使用哪个包下的Fragment

    相信很多朋友在使用Fragment的时候都会遇到下面这种情况:

    那么我们到底是使用android.app下的Fragment还是用的android.support.v4.app包下 的Fragment呢?

    其实都可以,前面说过Fragment是Android 3.0(API 11)后引入的,那么如果开发的app需要 在3.0以下的版本运行呢?比如还有一点点市场份额的2.3!于是乎,v4包就这样应运而生了, 而最低可以兼容到1.6版本!至于使用哪个包看你的需求了,现在3.0下手机市场份额其实已经不多了,随街都是4.0以上的,6.0十月份都出了,你说呢...所以这个时候,你可以直接使用app包下的Fragment 然后调用相关的方法,通常都是不会有什么问题的;如果你Fragment用了app包的, FragmentManager和FragmentTransaction都需要是app包的!要么用全部用app,要么全部用v4, 不然可是会报错的哦!当然如果你要自己的app对于低版本的手机也兼容的话,那么就可以选择用v4包!

    使用v4包下Fragment要注意的地方

    如果你使用了v4包下的Fragment,那么所在的那个Activity就要继承FragmentActivity哦! 案例:今天在xml文件中静态地载入fragment,然后重写了Fragment,但是在加载Activity的时候就报错了, 大概的提示就是Fragment错误还是找不到什么的,name属性改了几次还是错!最后才发现是用了 v4的包的缘故,只需让自己的Activity改成FragmentActivity即可!

    之前写了下面这段代码,然后报错: 

    5.1 Fragment基本概述

     有点莫名其妙啊,Fragment,FragmentManager,FragmentTransaction都是用的v4包啊, Activity也是继承FragmentActivity的啊?都改成app包就可以了,但是这不和我们用v4包的 前提冲突了么?其实也是有解决方法的哈?

    只需要把getFragmentManager( )改成getSupportFragmentManager( )就可以了

    创建一个Fragment

    静态加载Fragment

    定义Fragment的布局,就是fragment显示的内容

    自定义一个Fragment类,需要继承Fragment或者他的子类,重写onCreateView()方法 在该方法中调用:inflater.inflate()方法加载Fragment的布局文件,接着返回加载的view对象

    步骤一: 首先在MainActivity中绑定布局文件, 并且点击布局文件中的文字

    1. public class MainActivitys extends AppCompatActivity {
    2. @Override
    3. protected void onCreate(Bundle savedInstanceState) {
    4. super.onCreate(savedInstanceState);
    5. setContentView(R.layout.activity_main);
    6. //find views on onclick event
    7. findViewById(R.id.main_textView).setOnClickListener(v -> {
    8. // static load fragment.
    9. startActivity(new Intent(MainActivitys.this, StaticLoadFragmentActivity.class));
    10. });
    11. }
    12. }
    1. "1.0" encoding="utf-8"?>
    2. <LinearLayout
    3. xmlns:android="http://schemas.android.com/apk/res/android"
    4. xmlns:tools="http://schemas.android.com/tools"
    5. android:layout_width="match_parent"
    6. android:layout_height="match_parent"
    7. android:orientation="vertical"
    8. tools:context=".MainActivity2">
    9. <TextView
    10. android:id="@+id/main_textView"
    11. android:layout_width="wrap_content"
    12. android:layout_height="100dp"
    13. android:gravity="center"
    14. android:text="static load fragment"
    15. android:layout_marginBottom="@android:dimen/thumbnail_width"
    16. android:layout_marginLeft="@android:dimen/thumbnail_height"
    17. android:layout_marginTop="@android:dimen/thumbnail_height" />
    18. LinearLayout>

    步骤二: 再点击activety_main中的文字的时候加载StaticLoadFragmentActivity

    1. public class StaticLoadFragmentActivity extends AppCompatActivity {
    2. @Override
    3. protected void onCreate(@Nullable Bundle savedInstanceState) {
    4. super.onCreate(savedInstanceState);
    5. setContentView(R.layout.activity_static_load_fragment);
    6. }
    7. }

    在对应的布局文件activity_static_load_fragment, 导入真正要加载的那个fragment

    1. "1.0" encoding="utf-8"?>
    2. <LinearLayout
    3. xmlns:android="http://schemas.android.com/apk/res/android"
    4. xmlns:tools="http://schemas.android.com/tools"
    5. android:layout_width="match_parent"
    6. android:layout_height="match_parent"
    7. android:orientation="vertical"
    8. tools:context=".MainActivity">
    9. <fragment
    10. android:id="@+id/listFragment"
    11. android:name="com.example.testapplication.ListFragments"
    12. android:layout_width="300dp"
    13. android:layout_height="200dp" />
    14. LinearLayout>

    步骤三: 创建需要加载的fragment

    1. public class ListFragments extends Fragment {
    2. // 创建视图
    3. @Nullable
    4. @Override
    5. public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    6. View view = inflater.inflate(R.layout.fragment_test, container, false);
    7. TextView textView = view.findViewById(R.id.textView);
    8. textView.setText("Fragment test");
    9. return view;
    10. }
    11. }
    1. "1.0" encoding="utf-8"?>
    2. <RelativeLayout
    3. xmlns:android="http://schemas.android.com/apk/res/android"
    4. android:layout_width="match_parent"
    5. android:layout_height="match_parent"
    6. android:background="@color/cardview_dark_background">
    7. <TextView
    8. android:id="@+id/textView"
    9. android:layout_width="wrap_content"
    10. android:layout_height="wrap_content"
    11. android:layout_centerInParent="true"
    12. android:textColor="#FFFFFF"
    13. android:textSize="20sp"
    14. android:text="TextView" />
    15. RelativeLayout>

    动态加载Fragment

    动态加载Fragment的好处是加载比较灵活, 可以加任何的判断加什么fragment, 在哪里添加fragment

    步骤一: 在activity_main中创建container, 如下面代码中过的listContainer和detailContainer

    1. "1.0" encoding="utf-8"?>
    2. <LinearLayout
    3. xmlns:android="http://schemas.android.com/apk/res/android"
    4. xmlns:tools="http://schemas.android.com/tools"
    5. android:layout_width="match_parent"
    6. android:layout_height="match_parent"
    7. android:orientation="vertical"
    8. tools:context=".MainActivity2">
    9. <TextView
    10. android:id="@+id/main_textView"
    11. android:layout_width="match_parent"
    12. android:layout_height="100dp"
    13. android:gravity="center"
    14. android:text="static load fragment" />
    15. <LinearLayout
    16. android:orientation="horizontal"
    17. android:layout_width="match_parent"
    18. android:layout_height="match_parent">
    19. <LinearLayout
    20. android:orientation="horizontal"
    21. android:id="@+id/listContainer"
    22. android:layout_width="150dp"
    23. android:layout_margin="1dp"
    24. android:layout_height="match_parent">
    25. LinearLayout>
    26. <LinearLayout
    27. android:orientation="horizontal"
    28. android:id="@+id/detailContainer"
    29. android:layout_width="200dp"
    30. android:layout_margin="1dp"
    31. android:layout_height="match_parent">
    32. LinearLayout>
    33. LinearLayout>
    34. LinearLayout>

    步骤二: 创建fragment

    1. public class ListFragments extends Fragment {
    2. // 创建视图
    3. @Nullable
    4. @Override
    5. public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    6. View view = inflater.inflate(R.layout.fragment_test, container, false);
    7. return view;
    8. }
    9. }
    1. "1.0" encoding="utf-8"?>
    2. <RelativeLayout
    3. xmlns:android="http://schemas.android.com/apk/res/android"
    4. android:layout_width="match_parent"
    5. android:layout_height="match_parent"
    6. android:background="@color/cardview_dark_background">
    7. <TextView
    8. android:id="@+id/textView"
    9. android:layout_width="wrap_content"
    10. android:layout_height="wrap_content"
    11. android:layout_centerInParent="true"
    12. android:textColor="#FFFFFF"
    13. android:textSize="20sp"
    14. android:text="TextView" />
    15. RelativeLayout>

    步骤三: 通过FragmentManager将上面创建的fragment添加并且提交

    1. public class MainActivity extends AppCompatActivity {
    2. @Override
    3. protected void onCreate(Bundle savedInstanceState) {
    4. super.onCreate(savedInstanceState);
    5. setContentView(R.layout.activity_main);
    6. //find views on onclick event
    7. findViewById(R.id.main_textView).setOnClickListener(v -> {
    8. // static load fragment.
    9. startActivity(new Intent(MainActivity.this, StaticLoadFragmentActivity.class));
    10. });
    11. ListFragments listFragment = new ListFragments();
    12. getSupportFragmentManager()
    13. .beginTransaction()
    14. .add(R.id.listContainer, listFragment)
    15. .commit();
    16. ListFragments detailFragment = new ListFragments();
    17. getSupportFragmentManager()
    18. .beginTransaction()
    19. .add(R.id.detailContainer, detailFragment)
    20. .commit();
    21. // 移除fragment
    22. getSupportFragmentManager()
    23. .beginTransaction()
    24. .remove(detailFragment)
    25. .commit();
    26. // 替换fragment
    27. getSupportFragmentManager()
    28. .beginTransaction()
    29. .replace(R.id.detailContainer, listFragment)
    30. .commit();
    31. }
    32. }

    Fragment管理与Fragment事务

    Fragment与Activity的交互

    Activit传递数据给Fragment

    在Activity中创建Bundle数据包,调用Fragment实例的setArguments(bundle) 从而将Bundle数据包传给Fragment,然后Fragment中调用getArguments获得 Bundle对象,然后进行解析就可以了

    1. public class MainActivity extends AppCompatActivity {
    2. @Override
    3. protected void onCreate(Bundle savedInstanceState) {
    4. super.onCreate(savedInstanceState);
    5. setContentView(R.layout.activity_main);
    6. //find views on onclick event
    7. findViewById(R.id.main_textView).setOnClickListener(v -> {
    8. // static load fragment.
    9. startActivity(new Intent(MainActivity.this, StaticLoadFragmentActivity.class));
    10. });
    11. // activity ---> fragment value
    12. ListFragments listFragment = ListFragments.newInstance("list");
    13. getSupportFragmentManager()
    14. .beginTransaction()
    15. .add(R.id.listContainer, listFragment)
    16. .commit();
    17. ListFragments detailFragment = ListFragments.newInstance("detail");
    18. getSupportFragmentManager()
    19. .beginTransaction()
    20. .add(R.id.detailContainer, detailFragment)
    21. .commit();
    22. }
    23. }
    1. public class ListFragments extends Fragment {
    2. public static final String BUNDLE_TITLE = "bundle_title";
    3. private String mTitle = "sky";
    4. public static ListFragments newInstance(String title) {
    5. ListFragments fragments = new ListFragments();
    6. Bundle bundle = new Bundle();
    7. bundle.putString(BUNDLE_TITLE, title);
    8. fragments.setArguments(bundle);
    9. return fragments;
    10. }
    11. // 创建视图
    12. @Nullable
    13. @Override
    14. public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    15. View view = inflater.inflate(R.layout.fragment_test, container, false);
    16. TextView textView = view.findViewById(R.id.textView);
    17. textView.setText(mTitle);
    18. return view;
    19. }
    20. @Override
    21. public void onCreate(@Nullable Bundle savedInstanceState) {
    22. super.onCreate(savedInstanceState);
    23. // 在生命周期函数onCreate中获取这个参数
    24. if(getArguments() != null){
    25. mTitle = getArguments().getString(BUNDLE_TITLE);
    26. }
    27. }
    28. }

    如果传递的是一个对象, 那么可以用fragment对象调用需要传递的对象的set方法

    1. private User mUser;
    2. public void setUser(User user) {
    3. mUser = user;
    4. }
    5. public class User {
    6. }
    7. public static ListFragments newInstance(String title, User user){
    8. ListFragments fragments = new ListFragments();
    9. Bundle bundle = new Bundle();
    10. bundle.putString(BUNDLE_TITLE, title);
    11. fragments.setArguments(bundle);
    12. fragments.setUser(user);
    13. return fragments;
    14. }

    Fragment传递数据给Activity

    在Fragment中定义一个内部回调接口,再让包含该Fragment的Activity实现该回调接口, Fragment就可以通过回调接口传数据了

    步骤一: Fragment中定义一个回调接口并且接口回调

    1. public class ListFragments extends Fragment {
    2. public static final String BUNDLE_TITLE = "bundle_title";
    3. private String mTitle = "sky";
    4. public static ListFragments newInstance(String title) {
    5. ListFragments fragments = new ListFragments();
    6. Bundle bundle = new Bundle();
    7. bundle.putString(BUNDLE_TITLE, title);
    8. fragments.setArguments(bundle);
    9. return fragments;
    10. }
    11. // 创建视图
    12. @Nullable
    13. @Override
    14. public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    15. View view = inflater.inflate(R.layout.fragment_test, container, false);
    16. TextView textView = view.findViewById(R.id.textView);
    17. textView.setText(mTitle);
    18. // 点击textView, 调用回调方法
    19. textView.setOnClickListener(v -> {
    20. if(mOnTitleClickListener != null){
    21. mOnTitleClickListener.onClick(mTitle);
    22. }
    23. });
    24. return view;
    25. }
    26. @Override
    27. public void onCreate(@Nullable Bundle savedInstanceState) {
    28. super.onCreate(savedInstanceState);
    29. // 在生命周期函数onCreate中获取这个参数
    30. if(getArguments() != null){
    31. mTitle = getArguments().getString(BUNDLE_TITLE);
    32. }
    33. }
    34. // 设置接口的方法
    35. public void setOnTitleClickListener(OnTitleClickListener onTitleClickListener) {
    36. mOnTitleClickListener = onTitleClickListener;
    37. }
    38. // 定义变量
    39. private OnTitleClickListener mOnTitleClickListener;
    40. // 定义接口
    41. public interface OnTitleClickListener{
    42. void onClick(String title);
    43. }
    44. }

    步骤二:Activity中使用接口回调方法读数据

    1. public class MainActivity extends AppCompatActivity {
    2. @Override
    3. protected void onCreate(Bundle savedInstanceState) {
    4. super.onCreate(savedInstanceState);
    5. setContentView(R.layout.activity_main);
    6. //find views on onclick event
    7. findViewById(R.id.main_textView).setOnClickListener(v -> {
    8. // static load fragment.
    9. startActivity(new Intent(MainActivity.this, StaticLoadFragmentActivity.class));
    10. });
    11. // activity ---> fragment value
    12. ListFragments listFragment = ListFragments.newInstance("list");
    13. getSupportFragmentManager()
    14. .beginTransaction()
    15. .add(R.id.listContainer, listFragment)
    16. .commit();
    17. // fragment value ---> activity
    18. listFragment.setOnTitleClickListener(this::onClick);
    19. ListFragments detailFragment = ListFragments.newInstance("detail");
    20. getSupportFragmentManager()
    21. .beginTransaction()
    22. .add(R.id.detailContainer, detailFragment)
    23. .commit();
    24. // fragment value ---> activity
    25. detailFragment.setOnTitleClickListener(this::onClick);
    26. }
    27. public void onClick(String title) {
    28. setTitle(title); // 设置activity的标题
    29. }
    30. }

    总结下方法

    在Fragment定义一个接口,接口中定义抽象方法,你要传什么类型的数据参数就设置为什么类型;
    接着还有写一个调用接口中的抽象方法,把要传递的数据传过去
    再接着就是Activity了,调用Fragment提供的那个方法,然后重写抽象方法的时候进行数据 的读取就可以了!

    Fragment与Fragment之间的数据互传

    找到要接受数据的fragment对象,直接调用setArguments传数据进去就可以了 通常的话是replace时,即fragment跳转的时候传数据的,那么只需要在初始化要跳转的Fragment 后调用他的setArguments方法传入数据即可!
    如果是两个Fragment需要即时传数据,而非跳转的话,就需要先在Activity获得f1传过来的数据, 再传到f2了,就是以Activity为媒介~

    1. FragmentManager fManager = getSupportFragmentManager( );
    2. FragmentTransaction fTransaction = fManager.beginTransaction();
    3. Fragmentthree t1 = new Fragmentthree();
    4. Fragmenttwo t2 = new Fragmenttwo();
    5. Bundle bundle = new Bundle();
    6. bundle.putString("key",id);
    7. t2.setArguments(bundle);
    8. fTransaction.add(R.id.fragmentRoot, t2, "~~~");
    9. fTransaction.addToBackStack(t1);
    10. fTransaction.commit();

  • 相关阅读:
    Ubuntu20.04 配置 yolov5_ros 功能包记录
    脏牛提权 liunx
    《MATLAB 神经网络43个案例分析》:第36章 遗传算法优化计算——建模自变量降维
    面向对象设计原则之里氏代换原则
    LVS负载均衡集群
    大数据学习之Spark性能优化
    汽车行业调研:特种车市场发展趋势及现状分析
    UML建模语言分析和设计
    第四章·工厂方法模式
    kafka配置
  • 原文地址:https://blog.csdn.net/huangjianfeng21/article/details/132698070