Android-面试题Fragment详解

Posted 帅次

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android-面试题Fragment详解相关的知识,希望对你有一定的参考价值。

一、为什么要有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:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:background="@color/color_ff0000"
    android:gravity="center"
    >
    <TextView
        android:id="@+id/tv_on"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="首页"
        android:onClick="onClick"
        android:textColor="@color/white"
        android:textSize="@dimen/text_size_20"/>
</LinearLayout>

fragment_mine.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:background="@color/color_188FFF"
    android:gravity="center"
    >
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="我的"
        android:textColor="@color/white"
        android:textSize="@dimen/text_size_20"/>
</LinearLayout>

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

public class HomeFragment extends Fragment {
    private TextView tv_on;
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_home, container,false);
        MLog.e(this.getClass().getName()+"onCreateView");
        tv_on = view.findViewById(R.id.tv_on);
        tv_on.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                MLog.e(this.getClass().getName()+"准备关闭");
            }
        });
        return view;
    }
}

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

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/ll_bg"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/color_666666"
    android:orientation="horizontal">
    <fragment
        android:id="@+id/fragmen_home"
        android:name="com.scc.demo.fragment.HomeFragment"
        android:layout_width="0dp"
        android:layout_weight="1"
        android:layout_height="match_parent" />
    <fragment
        android:id="@+id/fragmen_mine"
        android:name="com.scc.demo.fragment.MineFragment"
        android:layout_width="0dp"
        android:layout_weight="1"
        android:layout_height="match_parent" />
</LinearLayout>

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

public class FragmentActivity extends ActivityBase {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        MLog.e(this.getClass().getName()+"onCreate");
        setContentView(R.layout.activity_fragment);
    }
}

动态创建

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

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

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

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

样例:

1.activity的布局文件

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <FrameLayout
        android:id="@+id/fl_frame"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_above="@+id/rg_bottom_tab"/>
    <RadioGroup
        android:id="@+id/rg_bottom_tab"
        android:layout_width="match_parent"
        android:layout_height="56dp"
        android:layout_alignParentBottom="true"
        android:background="#dcdcdc"
        android:orientation="horizontal">

        <RadioButton
            android:id="@+id/rb_home"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:button="@null"
            android:checked="true"
            android:gravity="center"
            android:text="首页"
            android:textSize="@dimen/text_size_18" />
        <RadioButton
            android:id="@+id/rb_list"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:button="@null"
            android:checked="true"
            android:gravity="center"
            android:text="列表"
            android:textSize="@dimen/text_size_18" />
        <RadioButton
            android:id="@+id/rb_news"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:button="@null"
            android:checked="true"
            android:gravity="center"
            android:text="消息"
            android:textSize="@dimen/text_size_18" />
        <RadioButton
            android:id="@+id/rb_mine"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            android:button="@null"
            android:gravity="center"
            android:text="我的"
            android:textSize="@dimen/text_size_18" />
    </RadioGroup>
</RelativeLayout>

2.自定义Fragment

package com.scc.demo.fragment;

import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import com.scc.demo.R;
import com.scc.demo.utils.MLog;

public class HomeFragment extends Fragment {
    private TextView tv_on;
    public static HomeFragment newInstance(String param){
        HomeFragment fragment = new HomeFragment();
        Bundle args = new Bundle();
        args.putString("param", param);
        fragment.setArguments(args);
        return fragment;
    }
    public HomeFragment(){}
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_home, container,false);
        MLog.e(this.getClass().getName()+"onCreateView");
        Bundle bundle = getArguments();
        String param = bundle.getString("param");
        tv_on = view.findViewById(R.id.tv_on);
        tv_on.setText("首页"+param);
        tv_on.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                MLog.e(this.getClass().getName()+"准备关闭");
            }
        });
        return view;
    }
}

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

package com.scc.demo.actvitiy;

import android.app.Fragment;
import android.app.FragmentManager;
import android.app.FragmentTransaction;
import android.os.Bundle;
import android.util.SparseArray;
import android.widget.RadioGroup;

import com.scc.demo.R;
import com.scc.demo.fragment.HomeFragment;
import com.scc.demo.fragment.ListFragment;
import com.scc.demo.fragment.MineFragment;
import com.scc.demo.fragment.NewsFragment;
import com.scc.demo.utils.MLog;

public class FragmentActivity extends ActivityBase {
    private RadioGroup rg_bottom_tab;
    private SparseArray<Fragment> mFragmentList;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_fragment);
        MLog.e(this.getClass().getName()+"onCreate");
        initView();
    }
    public void initView(){
        rg_bottom_tab = findViewById(R.id.rg_bottom_tab);
        mFragmentList = new SparseArray<>();
        mFragmentList.append(R.id.rb_home, HomeFragment.newInstance("我最帅"));
        mFragmentList.append(R.id.rb_list, ListFragment.newInstance("我最美"));
        mFragmentList.append(R.id.rb_news, NewsFragment.newInstance("我最新"));
        mFragmentList.append(R.id.rb_mine, MineFragment.newInstance("这里是我的"));
        rg_bottom_tab.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(RadioGroup group, int checkedId) {
                // 具体的fragment切换逻辑可以根据应用调整,例如使用show()/hide()
                FragmentManager fragmentManager = getFragmentManager();
                FragmentTransaction transaction = fragmentManager.beginTransaction();
                transaction.replace(R.id.fl_frame, mFragmentList.
                        get(checkedId));
                transaction.commit();
            }
        });
        // 默认显示第一个
        FragmentManager fragmentManager = getFragmentManager();
        FragmentTransaction transaction = fragmentManager.beginTransaction();
        transaction.add(R.id.fl_frame, mFragmentList.get(R.id.rb_home)).commit();
    }
}

四、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()。

 

以上是关于Android-面试题Fragment详解的主要内容,如果未能解决你的问题,请参考以下文章

Android面试题

Android面试题Android高级/资深面试题

Android 片段生命周期

Listview 项目未在 Fragment Android 中显示

Android从Fragment跳转Activty

RadioGroup结合RadioButton使用切换Fragment片段