Android通用的Adapter,或许你用我这一个就够了(2)

本文详细介绍了RecyclerView、ListView和GridView的适配器实现,包括ViewHolder的使用,以及通用适配器的设计。同时涵盖了热修复、插件化、组件化、图片加载、网络请求、RXJava响应式编程和依赖注入等Android开发的关键技术。
摘要由CSDN通过智能技术生成

@Override

public long getItemId(int position)

{

return position;

}

@Override

public View getView(int position, View convertView, ViewGroup parent)

{

ViewHolder viewHolder = null;

if (convertView == null)

{

convertView = mInflater.inflate(R.mLayoutId, parent, false);

viewHolder = new ViewHolder();

viewHolder.mTextView = (TextView) convertView

.findViewById(R.id.id_tv);

convertView.setTag(viewHolder);

} else

{

viewHolder = (ViewHolder) convertView.getTag();

}

viewHolder.mTextView.setText(mDatas.get(position));

return convertView;

}

private final class ViewHolder

{

TextView mTextView;

}

}

private class RecyclerAdapter extends RecyclerView.Adapter<RecyclerAdapter.Holder> {

private LayoutInflater mInflater;

private Context mContext;

private List mData;

private int mLayoutId;

public RecyclerAdapter(Context context, List data, int layoutId) {

mInflater = LayoutInflater.from(context);

this.mContext = context;

this.mData = data;

this.mLayoutId = layoutId;

}

@Override

public Holder onCreateViewHolder(ViewGroup parent, int viewType) {

View view = mInflater.inflate(mLayoutId, parent, false);

return new Holder(view);

}

@Override

public void onBindViewHolder(Holder holder, int position) {

holder.tv.setText(mData.get(position));

}

@Override

public int getItemCount() {

return mData.size();

}

class Holder extends RecyclerView.ViewHolder {

public TextView tv;

public Holder(View itemView) {

super(itemView);

tv = itemView.findViewById(R.id.tv);

}

}

}

三、ListView、RecyclerView Adapter一些共性分析
  1. 要显示的条目个数

  2. 要显示和条目对应的类型

  3. ListView、RecyclerView中,总共有多少个条目类型

  4. 根据不同的条目类型加载不同的View

  5. ListView复用条目convertView,RecyclerView自带ViewHoler复用

四、抽取RecyclerView、ListView、GridView适配器的基类

由于要通用RecyclerView、ListView、GridView只能将RecyclerView.Adapter和ListAdapter,SpinnerAdapter

这些都给实现了。里面也没多少个方法,其实也是很简单。至于一些方法的实现我考贝BaseAdapter的实现,

大家可以去看看BaseAdapter的源码。

public class BaseAdapter extends RecyclerView.Adapter implements ListAdapter, SpinnerAdapter {

private final DataSetObservable mDataSetObservable = new DataSetObservable();

// RecyclerView========================================================================

@Override

public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

return null;

}

// RecyclerView

@Override

public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {

}

// RecyclerView

@Override

public int getItemCount() {

return 0;

}

// RecyclerView========================================================================

// ListView========================================================================

@Override

public boolean areAllItemsEnabled() {

return true;

}

@Override

public boolean isEnabled(int position) {

return true;

}

@Override

public void registerDataSetObserver(DataSetObserver observer) {

mDataSetObservable.registerObserver(observer);

}

@Override

public void unregisterDataSetObserver(DataSetObserver observer) {

mDataSetObservable.unregisterObserver(observer);

}

public void notifyListDataSetChanged() {

mDataSetObservable.notifyChanged();

}

@Override

public Object getItem(int position) {

return null;

}

@Override

public View getView(int position, View convertView, ViewGroup parent) {

return null;

}

@Override

public int getViewTypeCount() {

return 1;

}

@Override

public int getItemViewType(int position) {

return 0;

}

@Override

public int getCount() {

return 0;

}

@Override

public boolean isEmpty() {

return getCount() == 0;

}

@Override

public View getDropDownView(int position, View convertView, ViewGroup parent) {

return getView(position, convertView, parent);

}

}

五、抽取支持ListView、 GridView、 RecyclerView多条目的共性

public interface QuickMultiSupport {

/**

  • 获取多条目View类型的数量

*/

int getViewTypeCount();

/**

  • 根据数据,获取多条目布局ID

*/

int getLayoutId(T data);

/**

  • 根据数据,获取多条目的ItemViewType

*/

int getItemViewType(T data);

/**

  • 是否合并条目–>>使用RecyclerView时,如果无效,请用原生的RecyclerView

*/

boolean isSpan(T data);

六、打造RecyclerView、ListView、GridView通用的适配器

由于要同时适配RecyclerView、ListView、GridView,也就等于把两个Adapter的实现放在一个类里面

所以代码有点多,但代码我已分开,很容易看。

还有数据都复制出来的,条目的增删改查都在Adapter里面。

/**

  • RecyclerView、ListView、GridView通用的适配器

*/

public abstract class QuickAdapter extends BaseAdapter {

private Context mContext;

private List mData;

private int mLayoutId;

private QuickMultiSupport mSupport;

private boolean isRecycler;

private int mPosition;

public QuickAdapter(Context context, List data, int layoutId) {

this.mContext = context;

this.mData = data == null ? new ArrayList() : new ArrayList(data);

this.mLayoutId = layoutId;

}

public QuickAdapter(Context context, List data, QuickMultiSupport support) {

this(context, data, 0);

this.mSupport = support;

}

@Override

public int getCount() {

return mData.size();

}

@Override

public T getItem(int position) {

return mData.get(position);

}

@Override

public long getItemId(int position) {

return position;

}

@Override

public View getView(int position, View convertView, ViewGroup parent) {

QuickViewHolder holder;

if (convertView == null) {

int layoutId = mLayoutId;

// 多条目的

if (mSupport != null) {

layoutId = mSupport.getLayoutId(mData.get(position));

}

// 创建ViewHolder

holder = createListHolder(parent, layoutId);

} else {

holder = (QuickViewHolder) convertView.getTag();

// 防止失误,还要判断

if (mSupport != null) {

int layoutId = mSupport.getLayoutId(mData.get(position));

// 如果布局ID不一样,又重新创建

if (layoutId != holder.getLayoutId()) {

// 创建ViewHolder

holder = createListHolder(parent, layoutId);

}

}

}

// 绑定View的数据

convert(holder, mData.get(position), position);

return holder.itemView;

}

/**

  • 创建ListView的Holer

*/

@NonNull

private QuickViewHolder createListHolder(ViewGroup parent, int layoutId) {

QuickViewHolder holder;

View itemView = LayoutInflater.from(mContext).inflate(layoutId, parent, false);

holder = new QuickViewHolder(itemView, layoutId);

itemView.setTag(holder);

return holder;

}

/**

  • ViewType的数量

*/

@Override

public int getViewTypeCount() {

// 多条目的

if (mSupport != null) {

return mSupport.getViewTypeCount() + super.getViewTypeCount();

}

return super.getViewTypeCount();

}

/**

  • 这个方法是共用的

*/

@Override

public int getItemViewType(int position) {

mPosition = position;

// 多条目的

if (mSupport != null) {

return mSupport.getItemViewType(mData.get(position));

}

return super.getItemViewType(position);

}

// RecyclerView=================================================================================

@Override

public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

isRecycler = true;

// 如果是多条目,viewType就是布局ID

View view;

if (mSupport != null) {

int layoutId = mSupport.getLayoutId(mData.get(mPosition));

view = LayoutInflater.from(mContext).inflate(layoutId, parent, false);

} else {

view = LayoutInflater.from(mContext).inflate(mLayoutId, parent, false);

}

QuickViewHolder holder = new QuickViewHolder(view);

return holder;

}

@Override

public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {

if (holder instanceof QuickViewHolder) {

convert((QuickViewHolder) holder, mData.get(position), position);

}

}

@Override

public int getItemCount() {

return mData.size();

}

@Override

public void onAttachedToRecyclerView(RecyclerView recyclerView) {

if (mSupport == null || recyclerView == null) {

return;

}

RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();

if (layoutManager instanceof GridLayoutManager) {

final GridLayoutManager gridLayoutManager = (GridLayoutManager) layoutManager;

final GridLayoutManager.SpanSizeLookup spanSizeLookup = gridLayoutManager.getSpanSizeLookup();

// 如果设置合并单元格就占用SpanCount那个多个位置

gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {

@Override

public int getSpanSize(int position) {

if (mSupport.isSpan(mData.get(position))) {

return gridLayoutManager.getSpanCount();

} else if (spanSizeLookup != null) {

return spanSizeLookup.getSpanSize(position);

}

return 1;

}

});

gridLayoutManager.setSpanCount(gridLayoutManager.getSpanCount());

}

}

@Override

public void onViewAttachedToWindow(RecyclerView.ViewHolder holder) {

if (mSupport == null) {

return;

}

int position = holder.getLayoutPosition();

// 如果设置合并单元格

if (mSupport.isSpan(mData.get(position))) {

ViewGroup.LayoutParams lp = holder.itemView.getLayoutParams();

if (lp != null && lp instanceof StaggeredGridLayoutManager.LayoutParams) {

StaggeredGridLayoutManager.LayoutParams p = (StaggeredGridLayoutManager.LayoutParams) lp;

p.setFullSpan(true);

}

}

}

// RecyclerView=================================================================================

/**

  • 绑定View的数据

*/

protected abstract void convert(QuickViewHolder holder, T item, int position);

//数据相关======

public void add(T elem) {

mData.add(elem);

notifyData();

}

public void addAll(List data) {

mData.addAll(data);

notifyData();

}

public void addFirst(T elem) {

mData.add(0, elem);

notifyData();

}

public void set(T oldElem, T newElem) {

set(mData.indexOf(oldElem), newElem);

notifyData();

}

public void set(int index, T elem) {

mData.set(index, elem);

notify();

}

public void remove(T elem) {

mData.remove(elem);

notifyData();

}

public void remove(int index) {

mData.remove(index);

notifyData();

}

public void replaceAll(List elem) {

mData.clear();

mData.addAll(elem);

notifyData();

}

/**

  • 清除

*/

public void clear() {

mData.clear();

notifyData();

}

private void notifyData() {

if (isRecycler) {

notifyDataSetChanged();

} else {

notifyListDataSetChanged();

}

}

public List getData() {

return mData;

}

}

七、通用ViewHolder的抽取

ViewHolder最主要就是实现平时常用的方法,比如:点击事件、长按事件、给TextView设置内容等等

如果不够,实际开发可以自己添加。

public class QuickViewHolder extends RecyclerView.ViewHolder {

private SparseArray<WeakReference> mViews;

private int mLayoutId;

public QuickViewHolder(View itemView) {

this(itemView, -1);

}

public QuickViewHolder(View itemView, int layoutId) {

super(itemView);

mViews = new SparseArray<>();

this.mLayoutId = layoutId;

}

public int getLayoutId() {

return mLayoutId;

}

/**

  • 设置条目的点击事件

*/

public QuickViewHolder setOnClickListener(View.OnClickListener listener) {

itemView.setOnClickListener(listener);

return this;

}

/**

  • 设置条目的长按事件

*/

public QuickViewHolder setOnLongClickListener(View.OnLongClickListener listener) {

itemView.setOnLongClickListener(listener);

return this;

}

/**

  • 设置View的点击事件

  • @return

*/

public QuickViewHolder setOnClickListener(int viewId, View.OnClickListener listener) {

View view = getView(viewId);

if (view != null) {

view.setOnClickListener(listener);

}

return this;

}

《设计思想解读开源框架》

第一章、 热修复设计

  • 第一节、 AOT/JIT & dexopt 与 dex2oat

  • 第二节、 热修复设计之 CLASS_ISPREVERIFIED 问题

  • 第三节、热修复设计之热修复原理

  • 第四节、Tinker 的集成与使用(自动补丁包生成)

    第二章、 插件化框架设计

  • 第一节、 Class 文件与 Dex 文件的结构解读

  • 第二节、 Android 资源加载机制详解

  • 第三节、 四大组件调用原理

  • 第四节、 so 文件加载机制

  • 第五节、 Android 系统服务实现原理

    第三章、 组件化框架设计

  • 第一节、阿里巴巴开源路由框——ARouter 原理分析

  • 第二节、APT 编译时期自动生成代码&动态类加载

  • 第三节、 Java SPI 机制

  • 第四节、 AOP&IOC

  • 第五节、 手写组件化架构

    第四章、图片加载框架

  • 第一节、图片加载框架选型

  • 第二节、Glide 原理分析

  • 第三节、手写图片加载框架实战

    第五章、网络访问框架设计

  • 第一节、网络通信必备基础

  • 第二节、OkHttp 源码解读

  • 第三节、Retrofit 源码解析

    第六章、 RXJava 响应式编程框架设计

  • 第一节、链式调用

  • 第二节、 扩展的观察者模式

  • 第三节、事件变换设计

  • 第四节、Scheduler 线程控制

    第七章、 IOC 架构设计

  • 第一节、 依赖注入与控制反转

  • 第二节、ButterKnife 原理上篇、中篇、下篇

  • 第三节、Dagger 架构设计核心解密

    第八章、 Android 架构组件 Jetpack

  • 第一节、 LiveData 原理

  • 第二节、 Navigation 如何解决 tabLayout 问题

  • 第三节、 ViewModel 如何感知 View 生命周期及内核原理

  • 第四节、 Room 架构方式方法

  • 第五节、 dataBinding 为什么能够支持 MVVM

  • 第六节、 WorkManager 内核揭秘

  • 第七节、 Lifecycles 生命周期


    本文包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…

    《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
    4563851)]

    第四章、图片加载框架

  • 第一节、图片加载框架选型

  • 第二节、Glide 原理分析

  • 第三节、手写图片加载框架实战

    [外链图片转存中…(img-fAl80waz-1714764563851)]

    第五章、网络访问框架设计

  • 第一节、网络通信必备基础

  • 第二节、OkHttp 源码解读

  • 第三节、Retrofit 源码解析

    [外链图片转存中…(img-D0XyEm7m-1714764563852)]

    第六章、 RXJava 响应式编程框架设计

  • 第一节、链式调用

  • 第二节、 扩展的观察者模式

  • 第三节、事件变换设计

  • 第四节、Scheduler 线程控制

    [外链图片转存中…(img-C0hhDV5c-1714764563853)]

    第七章、 IOC 架构设计

  • 第一节、 依赖注入与控制反转

  • 第二节、ButterKnife 原理上篇、中篇、下篇

  • 第三节、Dagger 架构设计核心解密

    [外链图片转存中…(img-8PYW4Vyb-1714764563854)]

    第八章、 Android 架构组件 Jetpack

  • 第一节、 LiveData 原理

  • 第二节、 Navigation 如何解决 tabLayout 问题

  • 第三节、 ViewModel 如何感知 View 生命周期及内核原理

  • 第四节、 Room 架构方式方法

  • 第五节、 dataBinding 为什么能够支持 MVVM

  • 第六节、 WorkManager 内核揭秘

  • 第七节、 Lifecycles 生命周期

    [外链图片转存中…(img-wlCypSVf-1714764563855)]
    本文包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…
    [外链图片转存中…(img-xu2j12a5-1714764563856)]
    《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 24
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值