Android优雅地处理按钮重复点击的几种方法

所属分类: 软件编程 / Android 阅读数: 90
收藏 0 赞 0 分享

App中,有很大一部分场景是点击按钮,向服务端提交数据,由于网络请求需要时间,用户很可能会多次点击,造成数据重复提交,造成各种莫名其妙的问题。

因此,防止按钮多次点击,是Android开发中一个很重要的技术手段。

以前的处理方式

网上查找到的,或者你可能会想到的方法大概有这些:

1.每个按钮点击事件中,记录点击时间,判断是否超过点击时间间隔

private long mLastClickTime = 0;
public static final long TIME_INTERVAL = 1000L;
private Button btTest;
private void initView() {
 btTest = findViewById(R.id.bt_test);
 btTest.setOnClickListener(new View.OnClickListener() {
  @Override
  public void onClick(View v) {
   long nowTime = System.currentTimeMillis();
   if (nowTime - mLastClickTime > TIME_INTERVAL) {
    // do something
    mLastClickTime = nowTime;
   } else {
    Toast.makeText(MainActivity.this, "不要重复点击", Toast.LENGTH_SHORT).show();
   }
  }
 });
}

这种方式,每个点击事件都需要写一个时间判断,重复代码很多。

2.封装一个点击事件,处理点击间隔判断

public abstract class CustomClickListener implements View.OnClickListener {
 private long mLastClickTime;
 private long timeInterval = 1000L;

 public CustomClickListener() {

 }

 public CustomClickListener(long interval) {
  this.timeInterval = interval;
 }

 @Override
 public void onClick(View v) {
  long nowTime = System.currentTimeMillis();
  if (nowTime - mLastClickTime > timeInterval) {
   // 单次点击事件
   onSingleClick();
   mLastClickTime = nowTime;
  } else {
   // 快速点击事件
   onFastClick();
  }
 }

 protected abstract void onSingleClick();
 protected abstract void onFastClick();
}

使用:

btTest.setOnClickListener(new CustomClickListener() {
 @Override
 protected void onSingleClick() {
  Log.d("xxx", "onSingleClick");
 }

 @Override
 protected void onFastClick() {
  Log.d("xxx", "onFastClick");
 }
});

相比于第一种方式,这种方法将重复点击的判断封装在CustomClickListener内部,外部无需处理时间判断,只需要实现点击方法即可。

3.利用RxAndroid处理重复点击

RxView.clicks(view)
 .throttleFirst(1, TimeUnit.SECONDS)
 .subscribe(new Consumer<Object>() {
  @Override
  public void accept(Object o) throws Exception {
   // do something
  }
  });

响应式地处理按钮点击,利用rxjava的操作符,来防止重复点击,相较于第1,2方案来说,此方法更为优雅一些。

思考一下:

这三种方法,不论哪一种,都对原有点击事件有很大的侵入性,要么你需要往Click事件中加方法,要么你需要替换整个Click事件,那么,有没有一种方式,可以在不改动原有逻辑的情况下,又能很好地处理按钮的重复点击呢?

 

更为优雅的处理方式

往同一类型的所有方法,都加上统一的处理逻辑,我们很快就能想到一个词: AOP ,没错, 面向切面编程

如何使用AOP来解决重复点击问题?

1.引入Aspectj

Android 上使用AOP编程,一般使用Aspectj这个库

站在巨人的肩膀上,沪江已经开源了Aspectj的Gradle插件,方便我们使用Aspectj

在项目根目录下的build.gradle中,添加依赖:

dependencies {
  ......
  classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.0'
}

在app或其他module目录下的build.gradle中,添加:

apply plugin: 'android-aspectjx'
dependencies {
 ......
 implementation 'org.aspectj:aspectjrt:1.8.9'
}

2.添加一个自定义注解

@Retention(RetentionPolicy.CLASS)
@Target(ElementType.METHOD)
public @interface SingleClick {
 /* 点击间隔时间 */
 long value() default 1000;
}

添加自定义注解的原因是,方便管理哪些方法使用了重复点击的AOP,同时可以在注解中传入点击时间间隔,更加灵活。

3.封装一个重复点击判断工具类

public final class XClickUtil {

 /**
  * 最近一次点击的时间
  */
 private static long mLastClickTime;
 /**
  * 最近一次点击的控件ID
  */
 private static int mLastClickViewId;

 /**
  * 是否是快速点击
  *
  * @param v 点击的控件
  * @param intervalMillis 时间间期(毫秒)
  * @return true:是,false:不是
  */
 public static boolean isFastDoubleClick(View v, long intervalMillis) {
  int viewId = v.getId();
  long time = System.currentTimeMillis();
  long timeInterval = Math.abs(time - mLastClickTime);
  if (timeInterval < intervalMillis && viewId == mLastClickViewId) {
   return true;
  } else {
   mLastClickTime = time;
   mLastClickViewId = viewId;
   return false;
  }
 }
}

4.编写Aspect AOP处理类

@Aspect
public class SingleClickAspect {
 private static final long DEFAULT_TIME_INTERVAL = 5000;

 /** 
  * 定义切点,标记切点为所有被@SingleClick注解的方法
  */
 @Pointcut("execution(@me.baron.test.annotation.SingleClick * *(..))")
 public void methodAnnotated() {}

 /** 
  * 定义一个切面方法,包裹切点方法
  */
 @Around("methodAnnotated()")
 public void aroundJoinPoint(ProceedingJoinPoint joinPoint) throws Throwable {
  // 取出方法的参数
  View view = null;
  for (Object arg : joinPoint.getArgs()) {
   if (arg instanceof View) {
    view = (View) arg;
    break;
   }
  }
  if (view == null) {
   return;
  }
  // 取出方法的注解
  MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
  Method method = methodSignature.getMethod();
  if (!method.isAnnotationPresent(SingleClick.class)) {
   return;
  }
  SingleClick singleClick = method.getAnnotation(SingleClick.class);
  // 判断是否快速点击
  if (!XClickUtil.isFastDoubleClick(view, singleClick.value())) {
   // 不是快速点击,执行原方法
   joinPoint.proceed();
  }
 }
}

使用方法

private void initView() {
 btTest = findViewById(R.id.bt_test);
 btTest.setOnClickListener(new View.OnClickListener() {
  @SingleClick
  @Override
  public void onClick(View v) {
   // do something
  }
 });
}

只需要一个注解,即完成了按钮的防止重复点击,其他所有工作交给编译器,代码清爽了很多有木有。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

Android网络编程之获取网络上的Json数据实例

这篇文章主要介绍了Android网络编程之获取网络上的Json数据实例,本文用完整的代码实例讲解了在Android中读取网络中Json数据的方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中的windowSoftInputMode属性详解

这篇文章主要介绍了Android中的windowSoftInputMode属性详解,本文对windowSoftInputMode的9个属性做了详细总结,需要的朋友可以参考下
收藏 0 赞 0 分享

Android网络编程之UDP通信模型实例

这篇文章主要介绍了Android网络编程之UDP通信模型实例,本文给出了服务端代码和客户端代码,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中使用ListView实现漂亮的表格效果

这篇文章主要介绍了Android中使用ListView实现漂亮的表格效果,本文用详细的代码实例创建了一个股票行情表格,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中刷新界面的二种方法

这篇文章主要介绍了Android中刷新界面的二种方法,本文使用Handler、postInvalidate两种方法实现界面刷新,需要的朋友可以参考下
收藏 0 赞 0 分享

Android SDK三种更新失败及其解决方法

这篇文章主要介绍了Android SDK三种更新失败及其解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(一)

Android3.0(API level 11)开始,Android设备不再需要专门的菜单键。随着这种变化,Android app应该取消对传统6项菜单的依赖。取而代之的是提供anction bar来提供基本的用户功能
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(二)

这次将继续上一篇文章没有讲完的Menu的学习,上下文菜单(Context menu)和弹出菜单(Popup menu)
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(三)

今天继续昨天没有讲完的Menu的学习,主要是Popup Menu的学习,需要的朋友可以参考下
收藏 0 赞 0 分享

Android显示网络图片实例

这篇文章主要介绍了Android显示网络图片的方法,以实例形式展示了Android程序显示网络图片的方法,非常具有实用价值,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多