为了账号安全,请及时绑定邮箱和手机立即绑定

Android RecyclerView的ViewHolder和Adapter的封装优化

标签:
Android

前一段时间,因为项目需要使用了RecyclerView,为了方便使用还进行封装了,详细参见此处:RecyclerView中Adapter和ViewHolder的封装 - DevWiki。

那样的封装有几个问题:

1.   ViewHolder的setData(M data)虽然方便设置数据,但是ViewHolder需要知晓数据类型。ViewHolder应该只用作View的缓存,而不应该知晓填充View的数据。

2.   BaseAdapter无法添加Header和Footer。

3.   点击事件耦合性较高。

基于以上几点,对BaseHolder和BaseAdapter进行修改优化。

BaseHolder的优化

相对于旧版的BaseHolder:

1.   新版的去除的数据类型的注入,使ViewHolder只用来缓存View。

2.   添加SparseArray,使之来缓存View。

3.   添加BaseHolder(View view)构造器,外部更方便控制View。

4.   保留getContext()方法,方便获取Context对象。

新版的BaseHolder代码如下:

[代码]java代码:

?

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

/**

 * 基础的ViewHolder<br>

 * ViewHolder只作View的缓存,不关心数据内容

 * Created by DevWiki on 2016/5/17.

 */

public class BaseHolder extends RecyclerView.ViewHolder   {

  

    private SparseArray<view   deep="7"> viewArray;

  

    /**

     * 构造ViewHolder

     * @param parent 父类容器

     * @param resId 布局资源文件id

     */

    public BaseHolder(ViewGroup   parent, @LayoutRes int resId) {

        super(LayoutInflater.from(parent.getContext()).inflate(resId,   parent, false));

        viewArray   = new SparseArray<>();

    }

  

    /**

     * 构建ViewHolder

     * @param view 布局View

     */

    public BaseHolder(View view) {

        super(view);

        viewArray   = new SparseArray<>();

    }

  

    /**

     * 获取布局中的View

     * @param viewId view的Id

     * @param <t> View的类型

     * @return view

     */

    protected <t   extends="" view="">T getView(@IdRes int viewId){

        View   view = viewArray.get(viewId);

        if (view   == null) {

            view   = itemView.findViewById(viewId);

            viewArray.put(viewId,   view);

        }

        return (T)   view;

    }

  

    /**

     * 获取Context实例

     * @return context

     */

    protected Context getContext()   {

        return itemView.getContext();

    }

}</t></t></view>

 

Adapter部分的优化

Adapter拆分为两个抽象类:AbsAdapter与BaseAdapter,其中:
AbsAdapter:封装了和ViewHolder和HeaderView,FooterView相关的方法。
BaseAdapter:继承AbsAdapter,封装了数据相关的方法。

各自聚焦于不同的方面,方面日后扩展。

AbsAdapter的代码如下:

[代码]java代码:

?

001

002

003

004

005

006

007

008

009

010

011

012

013

014

015

016

017

018

019

020

021

022

023

024

025

026

027

028

029

030

031

032

033

034

035

036

037

038

039

040

041

042

043

044

045

046

047

048

049

050

051

052

053

054

055

056

057

058

059

060

061

062

063

064

065

066

067

068

069

070

071

072

073

074

075

076

077

078

079

080

081

082

083

084

085

086

087

088

089

090

091

092

093

094

095

096

097

098

099

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

/**

 * RecyclerView.Adapter的扩展,包含headerView/footerView等

 * Created by DevWiki on 2016/7/13.

 */

  

public abstract class AbsAdapter<m, vh="" extends=""   baseholder=""> extends RecyclerView.Adapter<baseholder> {

  

    private static final String TAG   = "AbsAdapter";

  

    public static final int VIEW_TYPE_HEADER   = 1024;

    public static final int VIEW_TYPE_FOOTER   = 1025;

  

    protected View headerView;

    protected View footerView;

  

    protected Context context;

  

    public AbsAdapter(Context   context) {

        this.context   = context;

    }

  

    @Override

    public final BaseHolder   onCreateViewHolder(ViewGroup parent, int viewType) {

        if (viewType   == VIEW_TYPE_HEADER) {

            return   new BaseHolder(headerView);

        } else if   (viewType == VIEW_TYPE_FOOTER) {

            return   new BaseHolder(footerView);

        } else {

            return   createCustomViewHolder(parent, viewType);

        }

    }

  

    /**

     * 创建自定义的ViewHolder

     *

     * @param parent 父类容器

     * @param viewType view类型{@link #getItemViewType(int)}

     * @return ViewHolder

     */

    public abstract VH   createCustomViewHolder(ViewGroup parent, int viewType);

  

    @Override

    public final void onBindViewHolder(BaseHolder   holder, int position) {

        switch (holder.getItemViewType())   {

            case   VIEW_TYPE_HEADER:

            case   VIEW_TYPE_FOOTER:

                break;

            default:

                bindCustomViewHolder((VH)   holder, position);

                break;

        }

    }

  

    /**

     * 绑定自定义的ViewHolder

     *

     * @param holder   ViewHolder

     * @param position 位置

     */

    public abstract void bindCustomViewHolder(VH   holder, int position);

  

    /**

     * 添加HeaderView

     *

     * @param headerView 顶部View对象

     */

    public void addHeaderView(View   headerView) {

        if (headerView   == null) {

            Log.w(TAG,   "add the header view is null");

            return   ;

        }

        this.headerView   = headerView;

        notifyDataSetChanged();

    }

  

    /**

     * 移除HeaderView

     */

    public void removeHeaderView()   {

        if (headerView   != null) {

            headerView   = null;

            notifyDataSetChanged();

        }

    }

  

    /**

     * 添加FooterView

     *

     * @param footerView View对象

     */

    public void addFooterView(View   footerView) {

        if (footerView   == null) {

            Log.w(TAG,   "add the footer view is null");

            return;

        }

        this.footerView   = footerView;

        notifyDataSetChanged();

    }

  

    /**

     * 移除FooterView

     */

    public void removeFooterView()   {

        if (footerView   != null) {

            footerView   = null;

            notifyDataSetChanged();

        }

    }

  

    /**

     * 获取附加View的数量,包括HeaderView和FooterView

     *

     * @return 数量

     */

    public int getExtraViewCount()   {

        int extraViewCount   = 0;

        if (headerView   != null) {

            extraViewCount++;

        }

        if (footerView   != null) {

            extraViewCount++;

        }

        return extraViewCount;

    }

  

    /**

     * 获取顶部附加View数量,即HeaderView数量

     * @return 数量

     */

    public int getHeaderExtraViewCount()   {

        return headerView   == null ? 0 : 1;

    }

  

    /**

     * 获取底部附加View数量,即FooterView数量

     * @return 数量,0或1

     */

    public int getFooterExtraViewCount()   {

        return footerView   == null ? 0 : 1;

    }

  

    @Override

    public abstract long getItemId(int   position);

}</baseholder></m,>

 

BaseAdapter的代码如下:

[代码]java代码:

?

001

002

003

004

005

006

007

008

009

010

011

012

013

014

015

016

017

018

019

020

021

022

023

024

025

026

027

028

029

030

031

032

033

034

035

036

037

038

039

040

041

042

043

044

045

046

047

048

049

050

051

052

053

054

055

056

057

058

059

060

061

062

063

064

065

066

067

068

069

070

071

072

073

074

075

076

077

078

079

080

081

082

083

084

085

086

087

088

089

090

091

092

093

094

095

096

097

098

099

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

/**

 * 基础的Adapter

 *

 * Created by DevWiki on 2016/7/13.

 */

  

public abstract class BaseAdapter<m, vh=""   extends="" baseholder=""> extends AbsAdapter<m,   vh=""> {

  

    private List<m> dataList;

  

    public BaseAdapter(Context   context) {

        super(context);

        this.dataList   = new ArrayList<>();

    }

  

    public BaseAdapter(Context   context, List<m> list) {

        super(context);

        this.dataList   = new ArrayList<>();

        this.dataList.addAll(list);

    }

  

    /**

     * 填充数据,此操作会清除原来的数据

     *

     * @param list 要填充的数据

     * @return true:填充成功并调用刷新数据

     */

    public boolean fillList(List<m>   list) {

        dataList.clear();

        boolean   result = dataList.addAll(list);

        if (result)   {

            notifyDataSetChanged();

        }

        return result;

    }

  

    /**

     * 追加一条数据

     *

     * @param data 要追加的数据

     * @return true:追加成功并刷新界面

     */

    public boolean appendItem(M   data) {

        boolean   result = dataList.add(data);

        if (result)   {

            if   (getHeaderExtraViewCount() == 0) {

                notifyItemInserted(dataList.size()   - 1);

            }   else {

                notifyItemInserted(dataList.size());

            }

        }

        return result;

    }

  

    /**

     * 追加集合数据

     *

     * @param list 要追加的集合数据

     * @return 追加成功并刷新

     */

    public boolean appendList(List<m>   list) {

        boolean   result = dataList.addAll(list);

        if (result)   {

            notifyDataSetChanged();

        }

        return result;

    }

  

    /**

     * 在最顶部前置数据

     *

     * @param data 要前置的数据

     */

    public void proposeItem(M data)   {

        dataList.add(0,   data);

        if (getHeaderExtraViewCount()   == 0) {

            notifyItemInserted(0);

        } else {

            notifyItemInserted(getHeaderExtraViewCount());

        }

    }

  

    /**

     * 在顶部前置数据集合

     *

     * @param list 要前置的数据集合

     */

    public void proposeList(List<m>   list) {

        dataList.addAll(0,   list);

        notifyDataSetChanged();

    }

  

    @Override

    public long getItemId(int position)   {

        return position;

    }

  

    @Override

    public final int getItemViewType(int   position) {

        if (headerView   != null && position == 0) {

            return   VIEW_TYPE_HEADER;

        } else if   (footerView != null && position == dataList.size() +   getHeaderExtraViewCount()) {

            return   VIEW_TYPE_FOOTER;

        } else {

            return   getCustomViewType(position);

        }

    }

  

    /**

     * 获取自定义View的类型

     *

     * @param position 位置

     * @return View的类型

     */

    public abstract int getCustomViewType(int   position);

  

    @Override

    public int getItemCount() {

        return dataList.size()   + getExtraViewCount();

    }

  

    /**

     * 根据位置获取一条数据

     *

     * @param position View的位置

     * @return 数据

     */

    public M getItem(int position)   {

        if (headerView   != null && position == 0

                ||   position >= dataList.size() + getHeaderExtraViewCount()) {

            return   null;

        }

        return headerView   == null ? dataList.get(position) : dataList.get(position - 1);

    }

  

    /**

     * 根据ViewHolder获取数据

     *

     * @param holder   ViewHolder

     * @return 数据

     */

    public M getItem(VH holder) {

        return getItem(holder.getAdapterPosition());

    }

  

    public void updateItem(M data)   {

        int index   = dataList.indexOf(data);

        if (index   < 0) {

            return;

        }

        dataList.set(index,   data);

        if (headerView   == null) {

            notifyItemChanged(index);

        } else {

            notifyItemChanged(index   + 1);

        }

    }

  

    /**

     * 移除一条数据

     *

     * @param position 位置

     */

    public void removeItem(int position)   {

        if (headerView   == null) {

            dataList.remove(position);

        } else {

            dataList.remove(position   - 1);

        }

        notifyItemRemoved(position);

    }

  

    /**

     * 移除一条数据

     *

     * @param data 要移除的数据

     */

    public void removeItem(M data)   {

        int index   = dataList.indexOf(data);

        if (index   < 0) {

            return;

        }

        dataList.remove(index);

        if (headerView   == null) {

            notifyItemRemoved(index);

        } else {

            notifyItemRemoved(index   + 1);

        }

    }

}</m></m></m></m></m></m,></m,>

 

点击事件的优化

为了降低点击事件的耦合性,将点击事件从Adapter中移除,使用另外一种方式来实现。使用代码如下:

[代码]java代码:

?

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

recyclerView.addOnItemTouchListener(new RecyclerItemClickListener(recyclerView,

        new RecyclerItemClickListener.OnItemClickListener()   {

            @Override

            public   void onItemClick(View view, int position) {

                Person   person = singleAdapter.getItem(position);

                Toast.makeText(SingleActivity.this,   "click:" + person,

                        Toast.LENGTH_SHORT).show();

            }

  

            @Override

            public   void onItemLongClick(View view, int position) {

                Person   person = singleAdapter.getItem(position);

                Toast.makeText(SingleActivity.this,   "Long Click:" + person,

                        Toast.LENGTH_SHORT).show();

            }

        }));

 

使用RecyclerView的自带的方法addOnItemTouchListener()实现点击和长点击事件。实现代码如下:

[代码]java代码:

?

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

/**

 * 点击事件

 * Created by DevWiki on 2016/7/16.

 */

  

public class RecyclerItemClickListener implements RecyclerView.OnItemTouchListener   {

  

    private OnItemClickListener   clickListener;

    private GestureDetector   gestureDetector;

  

    public interface OnItemClickListener   {

        /**

         *   点击时回调

         *

         *   @param view 点击的View

         *   @param position 点击的位置

         */

        void onItemClick(View   view, int position);

  

        /**

         *   长点击时回调

         *

         *   @param view 点击的View

         *   @param position 点击的位置

         */

        void onItemLongClick(View   view, int position);

    }

  

    public RecyclerItemClickListener(final   RecyclerView recyclerView,

                                     OnItemClickListener   listener) {

        this.clickListener   = listener;

        gestureDetector   = new GestureDetector(recyclerView.getContext(),

                new   GestureDetector.SimpleOnGestureListener() {

                    @Override

                    public   boolean onSingleTapUp(MotionEvent e) {

                        return   true;

                    }

  

                    @Override

                    public   void onLongPress(MotionEvent e) {

                        View   childView = recyclerView.findChildViewUnder(e.getX(), e.getY());

                        if   (childView != null && clickListener != null) {

                            clickListener.onItemLongClick(childView,

                                    recyclerView.getChildAdapterPosition(childView));

                        }

                    }

                });

    }

  

    @Override

    public boolean onInterceptTouchEvent(RecyclerView   rv, MotionEvent e) {

        View   childView = rv.findChildViewUnder(e.getX(), e.getY());

        if (childView   != null && clickListener != null &&   gestureDetector.onTouchEvent(e)) {

            clickListener.onItemClick(childView,   rv.getChildAdapterPosition(childView));

            return   true;

        }

        return false;

    }

  

    @Override

    public void onTouchEvent(RecyclerView   rv, MotionEvent e) {

  

    }

  

    @Override

    public void onRequestDisallowInterceptTouchEvent(boolean   disallowIntercept) {

  

    }

}

 

分割线的优化

RecyclerView的分割线不像ListView那样方便,但是RecyclerView也提供了``方法来添加分割线。

分割线代码使用的是SDK中的Sample的代码,在其中添加了``方法,方便设置分割线的形式。

具体的代码参见此处:DividerDecoration.java

优化后的使用

优化后使用方式有所改变,具体使用方式如下:

[代码]java代码:

?

01

02

03

04

05

06

07

08

09

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

recyclerView = (RecyclerView)   findViewById(R.id.single_rv);

layoutManager = new LinearLayoutManager(this);

recyclerView.setLayoutManager(layoutManager);

DividerDecoration decoration = new DividerDecoration(this,   DividerDecoration.VERTICAL_LIST);

Drawable drawable =   getResources().getDrawable(R.drawable.divider_single);

decoration.setDivider(drawable);

recyclerView.addItemDecoration(decoration);

recyclerView.setAdapter(singleAdapter);

  

View view =   LayoutInflater.from(this).inflate(R.layout.item_single_header, null, false);

singleAdapter.addHeaderView(view);

  

recyclerView.addOnItemTouchListener(new RecyclerItemClickListener(recyclerView,

        new RecyclerItemClickListener.OnItemClickListener()   {

            @Override

            public   void onItemClick(View view, int position) {

                Person   person = singleAdapter.getItem(position);

                Toast.makeText(SingleActivity.this,   "click:" + person,

                        Toast.LENGTH_SHORT).show();

            }

  

            @Override

            public   void onItemLongClick(View view, int position) {

                Person   person = singleAdapter.getItem(position);

                Toast.makeText(SingleActivity.this,   "Long Click:" + person,

                        Toast.LENGTH_SHORT).show();

            }

        }));

 

以上所有的代码均在GitHub上,具体地址在此:Dev-Wiki/RecyclerView

原文链接:http://www.apkbus.com/blog-705730-61351.html

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消