Android RecyclerView的Item点击事件实现整理
发布时间 - 2026-01-10 22:37:51 点击率:次自从开始使用RecyclerView代替ListView,会发现有很多地方需要学习。前一段时间的学习记录有:

RecyclerView的滚动事件研究 - DevWiki
RecyclerView的ViewHolder和Adapter的封装优化 - DevWiki
RecyclerView问题记录 - DevWiki
实现 RecyclerView的Item的点击事件有三种方式:
- 在创建 ItemView时添加点击监听
- 当 ItemView attach RecyclerView时实现
- 通过RecyclerView已有的方法addOnItemTouchListener()实现
1.在创建ItemView时添加点击监听
思路是:因为ViewHolder我们可以拿到每个Item的根布局,所以如果我们为根布局设置单独的OnClick监听并将其开放给Adapter,那不就可以在组装RecyclerView时就能够设置ItemClickListener,只不过这个Listener不是设置到RecyclerView上而是设置到Adapter。具体实现代码如下:
public class SampleAdapter extends RecyclerView.Adapter<SampleAdapter.SampleViewHolder> {
private List<DataBean> mDatas;
private OnItemClickListener mListener; // Item点击事件
public DataBean getItem(int position) {
return mDatas == null ? null : mDatas.get(position);
}
@Override
public SampleViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent,false);
return new SampleViewHolder(itemView);
}
@Override
public void onBindViewHolder(SampleViewHolder holder, int position) {
}
@Override
public int getItemCount() {
return mDatas == null ? 0 : mDatas.size();
}
class SampleViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener, View.OnLongClickListener {
public SampleViewHolder(View itemView) {
super(itemView);
// TODO:初始化View
...
itemView.setOnClickListener(this);
itemView.setOnLongClickListener(this);
}
@Override
public void onClick(View v) {
if (mListener != null) {
mListener.onItemClick(SampleAdapter.this, v, getLayoutPosition());
}
}
@Override
public boolean onLongClick(View v) {
if (mListener != null) {
mListener.onItemLongClick(SampleAdapter.this, v, getLayoutPosition());
return true;
}
return false;
}
}
}
2.当ItemView attach RecyclerView时实现
该实现方法是在阅读国外的一篇博客时发现的,原文链接如下:Getting your clicks on RecyclerView
实现的代码如下:
public class ItemClickSupport {
private static final int KEY = 0x99999999;
private final RecyclerView mRecyclerView;
private OnItemClickListener mOnItemClickListener;
private OnItemLongClickListener mOnItemLongClickListener;
private View.OnClickListener mOnClickListener = new View.OnClickListener() {
@Override
public void onClick(View v) {
if (mOnItemClickListener != null) {
RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v);
mOnItemClickListener.onItemClicked(mRecyclerView, v, holder.getAdapterPosition());
}
}
};
private View.OnLongClickListener mOnLongClickListener = new View.OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
if (mOnItemLongClickListener != null) {
RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v);
return mOnItemLongClickListener.onItemLongClicked(mRecyclerView, v, holder.getAdapterPosition());
}
return false;
}
};
private RecyclerView.OnChildAttachStateChangeListener mAttachListener = new RecyclerView.OnChildAttachStateChangeListener() {
@Override
public void onChildViewAttachedToWindow(View view) {
if (mOnItemClickListener != null) {
view.setOnClickListener(mOnClickListener);
}
if (mOnItemLongClickListener != null) {
view.setOnLongClickListener(mOnLongClickListener);
}
}
@Override
public void onChildViewDetachedFromWindow(View view) {
}
};
/**
* ItemClickSupport的私有构造方法
*/
private ItemClickSupport(RecyclerView recyclerView) {
mRecyclerView = recyclerView;
mRecyclerView.setTag(KEY, this);
// 为RecyclerView设置OnChildAttachStateChangeListener事件监听
mRecyclerView.addOnChildAttachStateChangeListener(mAttachListener);
}
/**
* 为RecyclerView设置ItemClickSupport
*/
public static ItemClickSupport addTo(RecyclerView view) {
ItemClickSupport support = (ItemClickSupport) view.getTag(KEY);
if (support == null) {
support = new ItemClickSupport(view);
}
return support;
}
/**
* 为RecyclerView移除ItemClickSupport
*/
public static ItemClickSupport removeFrom(RecyclerView view) {
ItemClickSupport support = (ItemClickSupport) view.getTag(KEY);
if (support != null) {
support.detach(view);
}
return support;
}
/**
* 为RecyclerView设置点击事件监听
*/
public ItemClickSupport setOnItemClickListener(OnItemClickListener listener) {
mOnItemClickListener = listener;
return this;
}
/**
* 为RecyclerView设置长按事件监听
*/
public ItemClickSupport setOnItemLongClickListener(OnItemLongClickListener listener) {
mOnItemLongClickListener = listener;
return this;
}
/**
* 为RecyclerView移除OnChildAttachStateChangeListener事件监听
*/
private void detach(RecyclerView view) {
view.removeOnChildAttachStateChangeListener(mAttachListener);
view.setTag(KEY, null);
}
/**
* RecyclerView的点击事件监听接口
*/
public interface OnItemClickListener {
void onItemClicked(RecyclerView recyclerView, View itemView, int position);
}
/**
* RecyclerView的长按事件监听接口
*/
public interface OnItemLongClickListener {
boolean onItemLongClicked(RecyclerView recyclerView, View itemView, int position);
}
}
上面的代码中给RecyclerView设置了OnChildAttachStateChangeListener事件监听,当子View attach RecyclerView时设置事件监听。
private RecyclerView.OnChildAttachStateChangeListener mAttachListener = new RecyclerView.OnChildAttachStateChangeListener() {
@Override
public void onChildViewAttachedToWindow(View view) {
if (mOnItemClickListener != null) {
view.setOnClickListener(mOnClickListener);
}
if (mOnItemLongClickListener != null) {
view.setOnLongClickListener(mOnLongClickListener);
}
}
@Override
public void onChildViewDetachedFromWindow(View view) {}
};
使用时只需要调用addTo(RecycleView view)方法得到ItemClickSupport对象,然后调用setOnItemClickListener()方法和setOnItemLongClickListener()方法设置ItemView的点击事件和长按事件监听即可。
3.通过RecyclerView已有的方法addOnItemTouchListener()实现
3.1、查看源码
查看RecyclerView源码可以看到,RecyclerView预留了一个Item的触摸事件方法:
/**
* Add an {@link OnItemTouchListener} to intercept touch events before they are dispatched
* to child views or this view's standard scrolling behavior.
*
* <p>Client code may use listeners to implement item manipulation behavior. Once a listener
* returns true from
* {@link OnItemTouchListener#onInterceptTouchEvent(RecyclerView, MotionEvent)} its
* {@link OnItemTouchListener#onTouchEvent(RecyclerView, MotionEvent)} method will be called
* for each incoming MotionEvent until the end of the gesture.</p>
*
* @param listener Listener to add
* @see SimpleOnItemTouchListener
*/
public void addOnItemTouchListener(OnItemTouchListener listener) {
mOnItemTouchListeners.add(listener);
}
通过注释我们可知,此方法是在滚动事件之前调用,需要传入一个OnItemTouchListener对象。OnItemTouchListener的代码如下:
public static interface OnItemTouchListener {
public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e);
public void onTouchEvent(RecyclerView rv, MotionEvent e);
public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept);
}
此接口还提供了一个实现类,且官方推荐使用该实现类SimpleOnItemTouchListener:
/**
* An implementation of {@link RecyclerView.OnItemTouchListener} that has empty method bodies and
* default return values.
*
* You may prefer to extend this class if you don't need to override all methods. Another
* benefit of using this class is future compatibility. As the interface may change, we'll
* always provide a default implementation on this class so that your code won't break when
* you update to a new version of the support library.
*/
public static class SimpleOnItemTouchListener implements RecyclerView.OnItemTouchListener {
<span style="font-family:'Microsoft YaHei';">
</span> @Override
public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {
return false;
}
@Override
public void onTouchEvent(RecyclerView rv, MotionEvent e) {
}
@Override
public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {
}
}
在触摸接口中,当触摸时会回调一个MotionEvent对象,通过使用GestureDetectorCompat来解析用户的操作。
3.2、了解GestureDetector的工作原理
对于触摸屏,其原生的消息无非按下、抬起、移动这几种,我们只需要简单重载onTouch或者设置触摸侦听器setOnTouchListener即可进行处理。不过,为了提高我们的APP的用户体验,有时候我们需要识别用户的手势,Android给我们提供的手势识别工具GestureDetector就可以帮上大忙了。
GestureDetector的工作原理是,当我们接收到用户触摸消息时,将这个消息交给GestureDetector去加工,我们通过设置侦听器获得GestureDetector处理后的手势。
GestureDetector提供了两个侦听器接口,OnGestureListener处理单击类消息,OnDoubleTapListener处理双击类消息。
OnGestureListener的接口有这几个:
// 单击,触摸屏按下时立刻触发 abstract boolean onDown(MotionEvent e); // 抬起,手指离开触摸屏时触发(长按、滚动、滑动时,不会触发这个手势) abstract boolean onSingleTapUp(MotionEvent e); // 短按,触摸屏按下后片刻后抬起,会触发这个手势,如果迅速抬起则不会 abstract void onShowPress(MotionEvent e); // 长按,触摸屏按下后既不抬起也不移动,过一段时间后触发 abstract void onLongPress(MotionEvent e); // 滚动,触摸屏按下后移动 abstract boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY); // 滑动,触摸屏按下后快速移动并抬起,会先触发滚动手势,跟着触发一个滑动手势 abstract boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY); OnDoubleTapListener的接口有这几个:
// 双击,手指在触摸屏上迅速点击第二下时触发 abstract boolean onDoubleTap(MotionEvent e); // 双击的按下跟抬起各触发一次 abstract boolean onDoubleTapEvent(MotionEvent e); // 单击确认,即很快的按下并抬起,但并不连续点击第二下 abstract boolean onSingleTapConfirmed(MotionEvent e);
有时候我们并不需要处理上面所有手势,方便起见,Android提供了另外一个类SimpleOnGestureListener实现了如上接口,我们只需要继承SimpleOnGestureListener然后重载需要的手势即可。
3.3、实现点击事件监听
了解了GestureDetector的工作原理之后,便开始实现RecycleView的Item的点击事件。首先写一个SimpleRecycleViewItemClickListener类继承SimpleOnItemTouchListener,构造时传入Item点击回调OnItemClickListener,并覆写父类的boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e)方法,具体代码如下:
/**
* RecyclerView的Item点击事件监听
*
* @author liyunlong
* @date 2016/11/21 9:42
*/
public class SimpleRecycleViewItemClickListener extends RecyclerView.SimpleOnItemTouchListener {
private OnItemClickListener mListener;
private GestureDetectorCompat mGestureDetector;
public SimpleRecycleViewItemClickListener(OnItemClickListener listener) {
this.mListener = listener;
}
@Override
public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {
if (mGestureDetector == null) {
initGestureDetector(rv);
}
if (mGestureDetector.onTouchEvent(e)) { // 把事件交给GestureDetector处理
return true;
} else {
return false;
}
}
/**
* 初始化GestureDetector
*/
private void initGestureDetector(final RecyclerView recyclerView) {
mGestureDetector = new GestureDetectorCompat(recyclerView.getContext(), new GestureDetector.SimpleOnGestureListener() { // 这里选择SimpleOnGestureListener实现类,可以根据需要选择重写的方法
/**
* 单击事件
*/
@Override
public boolean onSingleTapUp(MotionEvent e) {
View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());
if (childView != null && mListener != null) {
mListener.onItemClick(childView, recyclerView.getChildLayoutPosition(childView));
return true;
}
return false;
}
/**
* 长按事件
*/
@Override
public void onLongPress(MotionEvent e) {
View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());
if (childView != null && mListener != null) {
mListener.onItemLongClick(childView, recyclerView.getChildLayoutPosition(childView));
}
}
/**
* 双击事件
*/
@Override
public boolean onDoubleTapEvent(MotionEvent e) {
int action = e.getAction();
if (action == MotionEvent.ACTION_UP) {
View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());
if (childView != null && mListener != null) {
mListener.onItemDoubleClick(childView, recyclerView.getChildLayoutPosition(childView));
return true;
}
}
return false;
}
});
}
/**
* RecyclerView的Item点击事件监听接口
*
* @author liyunlong
* @date 2016/11/21 9:43
*/
public interface OnItemClickListener {
/**
* 当ItemView的单击事件触发时调用
*/
void onItemClick(View view, int position);
/**
* 当ItemView的长按事件触发时调用
*/
void onItemLongClick(View view, int position);
/**
* 当ItemView的双击事件触发时调用
*/
void onItemDoubleClick(View view, int position);
}
/**
* RecyclerView的Item点击事件监听实现
*
* @author liyunlong
* @date 2016/11/21 10:05
*/
public class SimpleOnItemClickListener implements OnItemClickListener {
@Override
public void onItemClick(View view, int position) {
}
@Override
public void onItemLongClick(View view, int position) {
}
@Override
public void onItemDoubleClick(View view, int position) {
}
}
}
在GestureDetectorCompat的手势回调中我们覆写:
- boolean onSingleTapUp(MotionEvent e):单击事件回调
- void onLongPress(MotionEvent e):长按事件回调
- boolean onDoubleTapEvent(MotionEvent e):双击事件回调
如果我们只需要监听单击事件,而不需要监听长按事件和双击事件,构造SimpleRecycleViewItemClickListener时只需要传入SimpleOnItemClickListener即可,如果需要处理其它的手势监听,也可以覆写对应的手势回调方法。
4.三种方法对比
以上三种方式分别是:
- 在创建ItemView时添加点击监听
- 当ItemView attach RecyclerView时实现
- 通过RecyclerView已有的方法addOnItemTouchListener()实现
从以上三种方式的实现过程可知:
三种均可实现ItemView的点击事件和长按事件的监听。
第一种和第二种方式可以很方便对ItemView中的子View进行监听。
第三种方式可以很方便获取用户点击的坐标。
第二种方式和第三种方式可以写在单独的类中,相对于第一种写在Adapter的方式可使代码更独立整洁。
综上所述:
如果你只想监听ItemView的点击事件或长按事件,三种方式均可。
如果你想监听ItemView中每个子View的点击事件,采用第一种或者第二种比较方便。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# Android
# RecyclerView
# RecyclerView详解及实例
# item
# Android XRecyclerView最简单的item点击事件处理
# Recyclerview添加头布局和尾布局、item点击事件详解
# 使用RecyclerView实现Item点击事件
# 按下
# 双击
# 回调
# 单击
# 三种
# 只需要
# 是在
# 第二种
# 第一种
# 工作原理
# 均可
# 这几个
# 写在
# 很方便
# 移除
# 就可以
# 第三种
# 如果你
# 也不
# 有很多
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
JS中对数组元素进行增删改移的方法总结
如何用wdcp快速搭建高效网站?
使用spring连接及操作mongodb3.0实例
Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全
ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集
Laravel如何使用API Resources格式化JSON响应_Laravel数据资源封装与格式化输出
香港服务器部署网站为何提示未备案?
Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程
Laravel如何为API编写文档_Laravel API文档生成与维护方法
如何快速搭建安全的FTP站点?
百度输入法ai面板怎么关 百度输入法ai面板隐藏技巧
Laravel模型事件有哪些_Laravel Model Event生命周期详解
Laravel Telescope怎么调试_使用Laravel Telescope进行应用监控与调试
JS经典正则表达式笔试题汇总
Laravel如何处理表单验证?(Requests代码示例)
Swift中循环语句中的转移语句 break 和 continue
瓜子二手车官方网站在线入口 瓜子二手车网页版官网通道入口
青岛网站建设如何选择本地服务器?
如何确认建站备案号应放置的具体位置?
如何彻底删除建站之星生成的Banner?
详解Nginx + Tomcat 反向代理 负载均衡 集群 部署指南
公司门户网站制作流程,华为官网怎么做?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
如何在HTML表单中获取用户输入并结合JavaScript动态控制复利计算循环
非常酷的网站设计制作软件,酷培ai教育官方网站?
Python高阶函数应用_函数作为参数说明【指导】
简单实现jsp分页
如何用ChatGPT准备面试 模拟面试问答与职场话术练习教程
北京专业网站制作设计师招聘,北京白云观官方网站?
JS弹性运动实现方法分析
Laravel如何与Docker(Sail)协同开发?(环境搭建教程)
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
使用PHP下载CSS文件中的所有图片【几行代码即可实现】
Linux后台任务运行方法_nohup与&使用技巧【技巧】
如何用低价快速搭建高质量网站?
电商网站制作价格怎么算,网上拍卖流程以及规则?
弹幕视频网站制作教程下载,弹幕视频网站是什么意思?
Mybatis 中的insertOrUpdate操作
如何用虚拟主机快速搭建网站?详细步骤解析
如何用PHP工具快速搭建高效网站?
用yum安装MySQLdb模块的步骤方法
Laravel如何实现一对一模型关联?(Eloquent示例)
高防网站服务器:DDoS防御与BGP线路的AI智能防护方案
如何在阿里云通过域名搭建网站?
JavaScript如何实现路由_前端路由原理是什么
Laravel怎么清理缓存_Laravel optimize clear命令详解
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
Laravel N+1查询问题如何解决_Eloquent预加载(Eager Loading)优化数据库查询

