Basefragment的封装使用

1、前言

封装作为面向对象的三大特性之一,在我们平时的coding过程用的那是非常的多的。大多数时候,我们可以根据封装情况评判代码的质量水平。

本文主要讲解的是fragment的封装。大体上,在项目的初期,我们都会在基类中把一些常用的东西封装起来,以供子类方便的调用。这用做的好处是显而易见的:

  • 将一些共用的东西集中放在base基类中
  • 方便后期修改维护
  • 避免子类代码冗余
  • 消除不必要的重复代码
  • 方便子类的使用

2、开始封装

2.1 获取mActivity

在fragment中,我们可以通过getActivity()方法获取到当前依附的activity实例。但是如果在使用的时候直接获取有时候可能会报空指针,那么我们可以在fragment生命周期的onAttach(Context context)方法中获取到并提升为全局变量。

  /**
     * 贴附的activity
     */
    protected FragmentActivity mActivity;

    @Override
    public void onAttach(Context context)
    {
        super.onAttach(context);

        mActivity = getActivity();
    }

2.2 onCreateView方法的封装

通常,我们需要在fragment的onCreateView方法中做一些初始化的事情,比如加载布局文件,findview,网络加载数据,初始化数据等,将这些事情都写在一个方法中显得onCreateView非常的庞大,我们可以分成几个方法抽离出来,这样代码既美观又方便阅读:

@Override
    @Nullable
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
            @Nullable Bundle savedInstanceState)
    {
        mRootView = inflater.inflate(setLayoutResouceId(), container, false);

        initData(getArguments());

        initView();

        mIsPrepare = true;

        onLazyLoad();

        setListener();

        return mRootView;
    }

以上代码我们根据功能进行以下封装,注意方法调用顺序不可调换:

  • setLayoutResouceId():设置布局资源id
  • initData(getArguments()) :数据初始化,主要包括拆包从activity传递过来的参数,适配器初始化,集合初始化等,不可进行view的操作
  • mIsPrepare置为true:mIsPrepare 表示是否已经加载view,该属性主要用于懒加载(下文会详细介绍懒加载)
  • onLazyLoad():懒加载数据,在oncreatview方法中调用可以直接理解为加载数据,方法中可以进行view的操作
  • setListener():各种监听事件的统一设置

2.3 field介绍

封装的属性代码注释已经写的很清楚了:

  /**
     * 贴附的activity
     */
    protected FragmentActivity mActivity;

    /**
     * 根view
     */
    protected View mRootView;

    /**
     * 是否对用户可见
     */
    protected boolean mIsVisible;
    /**
     * 是否加载完成
     * 当执行完oncreatview,View的初始化方法后方法后即为true
     */
    protected boolean mIsPrepare;

2.4 封装的方法

以下封装的三个方法用于子类的实现:

   /**
     * 初始化数据
     * @author 漆可
     * @date 2016-5-26 下午3:57:48  
     * @param arguments 接收到的从其他地方传递过来的参数
     */
    protected void initData(Bundle arguments)
    {

    }

    /**
     * 初始化View
     * @author 漆可
     * @date 2016-5-26 下午3:58:49
     */
    protected void initView()
    {

    }

    /**
     * 设置监听事件
     * @author 漆可
     * @date 2016-5-26 下午3:59:36
     */
    protected void setListener()
    {

    }

2.5 setLayoutResouceId方法

setLayoutResouceId()是一个抽象方法,子类必须实现,用于获取布局文件的资源id,加载到fragment中,并在onCreatview方法中用mRootView保存其引用

 /**
     * 设置根布局资源id
     * @author 漆可
     * @date 2016-5-26 下午3:57:09  
     * @return
     */
    protected abstract int setLayoutResouceId();

2.6 打造专属的findViewById()方法

是不是很羡慕activity的findViewById()方法,在我们封装的basefragment中,也有自己的findViewById()方法,而且比activity的更加强大,因为它可以直接转化成我们相对应的view,而不需要强制转化。有了它,我们的fragment子类就可以直接使用了。

   @SuppressWarnings("unchecked")
    protected <T extends View> T findViewById(int id)
    {
        if (mRootView == null)
        {
            return null;
        }

        return (T) mRootView.findViewById(id);
    }

2.7 懒加载

何为懒加载,就是view没有与用户交互的话并不会加载,但是他的加载顺序又非常快。该方法主要在viewpager嵌套fragment中使用得多。我们都知道,viewpager可以提前加载左右指定数目(当然这个数目可以通过setOffscreenPageLimit(int limit)设置)的fragment,如果我们使用懒加载,就只会做些view的创建等操作,避免提前执行其他页面的网络请求。

一下代码中,setUserVisibleHint(boolean isVisibleToUser)表示是否与用户可交互,他在onCreatview方法前执行,当isVisibleToUser为true时表示对用户可见,在这里我们执行自定义的onVisibleToUser() 
方法,在onVisibleToUser()中,我们进行判断,当mIsPrepare为true且与用户可交互时执行我们的onLazyLoad()方法进行懒加载。

  @Override
    public void setUserVisibleHint(boolean isVisibleToUser)
    {
        super.setUserVisibleHint(isVisibleToUser);

        this.mIsVisible = isVisibleToUser;

        if (isVisibleToUser)
        {
            onVisibleToUser();
        }
    }

    /**
     * 用户可见时执行的操作
     * @author 漆可
     * @date 2016-5-26 下午4:09:39
     */
    protected void onVisibleToUser()
    {
        if (mIsPrepare && mIsVisible)
        {
            onLazyLoad();
        }
    }

3 结束

好了,整个basefragment的封装就做好了,下面是完整代码,由于代码量不多,本篇博客就不上传完整demo了。

public abstract class BaseFragment extends Fragment
{
    /**
     * 贴附的activity
     */
    protected FragmentActivity mActivity;

    /**
     * 根view
     */
    protected View mRootView;

    /**
     * 是否对用户可见
     */
    protected boolean mIsVisible;
    /**
     * 是否加载完成
     * 当执行完oncreatview,View的初始化方法后方法后即为true
     */
    protected boolean mIsPrepare;

    @Override
    public void onAttach(Context context)
    {
        super.onAttach(context);

        mActivity = getActivity();
    }

    @Override
    @Nullable
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
            @Nullable Bundle savedInstanceState)
    {
        mRootView = inflater.inflate(setLayoutResouceId(), container, false);

        initData(getArguments());

        initView();

        mIsPrepare = true;

        onLazyLoad();

        setListener();

        return mRootView;
    }

    /**
     * 初始化数据
     * @author 漆可
     * @date 2016-5-26 下午3:57:48  
     * @param arguments 接收到的从其他地方传递过来的参数
     */
    protected void initData(Bundle arguments)
    {

    }

    /**
     * 初始化View
     * @author 漆可
     * @date 2016-5-26 下午3:58:49
     */
    protected void initView()
    {

    }

    /**
     * 设置监听事件
     * @author 漆可
     * @date 2016-5-26 下午3:59:36
     */
    protected void setListener()
    {

    }

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser)
    {
        super.setUserVisibleHint(isVisibleToUser);

        this.mIsVisible = isVisibleToUser;

        if (isVisibleToUser)
        {
            onVisibleToUser();
        }
    }

    /**
     * 用户可见时执行的操作
     * @author 漆可
     * @date 2016-5-26 下午4:09:39
     */
    protected void onVisibleToUser()
    {
        if (mIsPrepare && mIsVisible)
        {
            onLazyLoad();
        }
    }

    /**
     * 懒加载,仅当用户可见切view初始化结束后才会执行
     * @author 漆可
     * @date 2016-5-26 下午4:10:20
     */
    protected void onLazyLoad()
    {

    }

    @SuppressWarnings("unchecked")
    protected <T extends View> T findViewById(int id)
    {
        if (mRootView == null)
        {
            return null;
        }

        return (T) mRootView.findViewById(id);
    }

    /**
     * 设置根布局资源id
     * @author 漆可
     * @date 2016-5-26 下午3:57:09  
     * @return
     */
    protected abstract int setLayoutResouceId();
}
原文地址:https://www.cnblogs.com/zhujiabin/p/5632112.html