App中,有很大一部分场景是点击按钮,向服务端提交数据,由于网络请求需要时间,用户很可能会多次点击,造成数据重复提交,造成各种莫名其妙的问题。
因此,防止按钮多次点击,是Android开发中一个很重要的技术手段。
网上查找到的,或者你可能会想到的方法大概有这些:
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(); } } }); } 复制代码
这种方式,每个点击事件都需要写一个时间判断,重复代码很多。
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内部,外部无需处理时间判断,只需要实现点击方法即可。
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 ,没错,面向切面编程。
Android 上使用AOP编程,一般使用Aspectj这个库
站在巨人的肩膀上,沪江已经开源了Aspectj的Gradle插件,方便我们使用Aspectj
dependencies { ...... classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.0' } 复制代码
// 注意:主App中请确保添加aspectjx apply plugin: 'android-aspectjx' dependencies { ...... implementation 'org.aspectj:aspectjrt:1.8.9' } 复制代码
@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface SingleClick { /* 点击间隔时间 */ long value() default 1000; } 复制代码
添加自定义注解的原因是,方便管理哪些方法使用了重复点击的AOP,同时可以在注解中传入点击时间间隔,更加灵活。
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; } } } 复制代码
@Aspect public class SingleClickAspect { private static final long DEFAULT_TIME_INTERVAL = 5000; /** * 定义切点,标记切点为所有被@SingleClick注解的方法 * 注意:这里me.baron.test.annotation.SingleClick需要替换成 * 你自己项目中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(2000) @SingleClick @Override public void onClick(View v) { // do something } }); } 复制代码
只需要一个注解,即完成了按钮的防止重复点击,其他所有工作交给编译器,代码清爽了很多有木有。