Android開(kāi)發(fā)一行代碼解決安卓重復(fù)點(diǎn)擊
拋出問(wèn)題
“大哥,有個(gè)問(wèn)題想問(wèn)你!”
“哎,說(shuō)吧(內(nèi)心戲:咋又來(lái)了。。。準(zhǔn)沒(méi)好事?。?rdquo;
“我的一個(gè)頁(yè)面中有一個(gè)查詢按鈕,點(diǎn)擊就會(huì)發(fā)出網(wǎng)絡(luò)請(qǐng)求,等待返回結(jié)果后更新數(shù)據(jù)。”
“這不挺好的嘛!有啥問(wèn)題啊?”
“對(duì),我也覺(jué)得沒(méi)問(wèn)題,但測(cè)試不按套路出牌啊,測(cè)試那邊的網(wǎng)絡(luò)不太好,她點(diǎn)擊按鈕之后由于網(wǎng)絡(luò)比較慢就快速多點(diǎn)擊了幾下,然后。。。”
“然后怎么了?ANR了吧?”
“你咋知道的大哥?”
“來(lái)吧,幫您看看吧!”
日常開(kāi)發(fā)中肯定遇到過(guò)這種情況,接下來(lái)咱們就來(lái)看看該怎么解決這種問(wèn)題。
第一種:彈窗等待
“小子,過(guò)來(lái),你看啊,你可以這樣,當(dāng)你點(diǎn)擊了按鈕之后就彈出一個(gè)對(duì)話框,設(shè)置成不能關(guān)閉,等網(wǎng)絡(luò)請(qǐng)求完成之后再將對(duì)話框關(guān)閉了就行”
“這是一種方式,但我該怎么寫(xiě)呢?”
“哎,給你寫(xiě)一下例子吧:照著下面代碼寫(xiě)就行”
public void btnDialog(View view) {
ProgressDialog progressDialog = new ProgressDialog(this);
progressDialog.setTitle("等待");
progressDialog.setMessage("等待內(nèi)容");
//progressDialog.setCanceledOnTouchOutside(false);
progressDialog.show();
}
上面代碼很簡(jiǎn)單,這只是一種思路,點(diǎn)擊按鈕后可以彈出對(duì)話框不讓用戶進(jìn)行操作(注釋的那一行代碼就是禁止用戶點(diǎn)擊的),當(dāng)請(qǐng)求完成之后再將對(duì)話框關(guān)閉。
不過(guò)不推薦這種做法,官方也不推薦,官方推薦使用ProgressBar。
第二種:禁止點(diǎn)擊
“大哥,我覺(jué)得彈出對(duì)話框不太好,會(huì)讓用戶很反感,還有別的方式嗎?”
“行了,早就準(zhǔn)備好和你說(shuō)了,還不止一種呢!這種方式更簡(jiǎn)單了,只需要設(shè)置按鈕是否可以點(diǎn)擊就行,當(dāng)用戶點(diǎn)擊后設(shè)置按鈕不可點(diǎn)擊,請(qǐng)求完成之后再設(shè)置可以點(diǎn)擊就行了,這個(gè)不用我寫(xiě)代碼了吧?“
”嘿嘿,這個(gè)不用,你剛才說(shuō)還有好幾種,說(shuō)來(lái)聽(tīng)聽(tīng)??!“
第三種:時(shí)間判斷
這種方式比上面的稍微麻煩點(diǎn),但還是很簡(jiǎn)單。
具體操作就是定義兩個(gè)變量,一個(gè)為上次點(diǎn)擊時(shí)間,一個(gè)為點(diǎn)擊的間隔時(shí)間。
// 上次點(diǎn)擊時(shí)間 private long lastClickTime = 0L; // 兩次點(diǎn)擊間隔時(shí)間(毫秒) private static final int FAST_CLICK_DELAY_TIME = 1500;
在點(diǎn)擊時(shí)進(jìn)行判斷就可以了,來(lái)看一下代碼吧:
public void btnInter(View view) {
if (System.currentTimeMillis() - lastClickTime >= FAST_CLICK_DELAY_TIME)
lastClickTime = System.currentTimeMillis();
}
}
"大哥,這種方法看著比上面兩種好用多了,只需要這樣定義一下就行了,我就用這一種?。?quot;
“先別高興的太早了!”
第四種:AOP實(shí)現(xiàn)
“大哥,你剛才說(shuō)我高興的太早了是為啥?。?ldquo;
”你只有一個(gè)頁(yè)面的話這樣寫(xiě)肯定是沒(méi)有問(wèn)題的,但是如果有多個(gè)頁(yè)面都有防止按鈕重復(fù)點(diǎn)擊的需求呢?每個(gè)頁(yè)面都定義一遍啊?“
”呃呃,你說(shuō)的對(duì),大哥,那應(yīng)該怎么辦呢?“
”你知道AOP嗎?接下來(lái)我要說(shuō)的就和它有關(guān)“
”AOP?那是什么鬼?我知道OOP!“
”不錯(cuò)啊小子,還知道OOP,OOP就是面向?qū)ο螅鳤OP就是面向過(guò)程。“
AOP為Aspect OrientedProgramming的縮寫(xiě),意為面向切面編程。所謂的面向切面編程其實(shí)是對(duì)業(yè)務(wù)邏輯又進(jìn)行了進(jìn)一步的抽取,將多種業(yè)務(wù)邏輯中的公用部分抽取出來(lái)做成一種服務(wù)(比如日志記錄,性能統(tǒng)計(jì)等),從而實(shí)現(xiàn)代碼復(fù)用。另外這種服務(wù)通過(guò)配置可以動(dòng)態(tài)的給程序添加統(tǒng)一控制,利用AOP可以對(duì)業(yè)務(wù)邏輯的各個(gè)部分進(jìn)行分離,從而使得業(yè)務(wù)邏輯各部分之間的耦合度降低。
AOP并不是Android中的產(chǎn)物,而是Java中的,Android官方并沒(méi)有提供,所以想使用AOP首先要導(dǎo)入可以實(shí)現(xiàn)AOP的三方庫(kù):
在項(xiàng)目級(jí)別的build.gradle中新增以下代碼:
classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.4'
然后在moudle的build.gradle中新增依賴:
implementation 'org.aspectj:aspectjrt:1.8.14'
還需要在moudle的build.gradle中最上面添加以下代碼:
apply plugin: 'android-aspectjx'
完事之后點(diǎn)擊sync Now更新一下。
因?yàn)橄朐谒械胤蕉寄苁褂?,所以?lái)定義一個(gè)注解:
/**
* 實(shí)現(xiàn)防止按鈕連續(xù)點(diǎn)擊
* @author jiang zhu on 2020/4/19
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface SingleClick {
/* 點(diǎn)擊間隔時(shí)間 */
long value() default 1500;
}
注解大家應(yīng)該都使用過(guò),運(yùn)行時(shí)就不說(shuō)了,使用范圍定義的是方法上,有一個(gè)參數(shù)為間隔時(shí)間,這個(gè)參數(shù)意思其實(shí)和第三種方案中的類似。
然后就需要定義一個(gè)類,里面實(shí)現(xiàn)AOP的具體操作:
@Aspect
public class SingleClickAspect {}
然后定義切點(diǎn),標(biāo)記切點(diǎn)為所有被@SingleClick的方法:
@Pointcut("execution(@com.zj.singclick.SingleClick * *(..))")
public void methodAnnotated() {
}
這里要注意包名和類名一定要寫(xiě)對(duì)。
接下來(lái)定義一個(gè)切面方法,包裹著切點(diǎn)方法:
@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 (!isFastDoubleClick(view, singleClick.value())) {
joinPoint.proceed();
}
}
來(lái)簡(jiǎn)單分析下,首先看注解中定義了剛才定義的切點(diǎn)方法,然后取出方法的參數(shù),再取出方法的注解,然后調(diào)用
isFastDoubleClick方法判斷是否為快速點(diǎn)擊事件,如果是什么都不干,如果不是則執(zhí)行原方法中的內(nèi)容。
下面貼一下isFastDoubleClick方法的代碼:
private static long mLastClickTime;
private static int mLastClickViewId;
private 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;
}
}
上面代碼采用了View的id和間隔時(shí)間雙重判斷,即使一個(gè)頁(yè)面的多個(gè)按鈕都可區(qū)分。
“大哥,停停停,都給我說(shuō)懵了,這寫(xiě)完這個(gè)該怎么用?。?rdquo;
“別著急,下面就教你該怎么使用!”
使用很簡(jiǎn)單,只需要在點(diǎn)擊時(shí)間方法上加上咱們自定義的注解就行了,還可以設(shè)置間隔時(shí)間,如果不寫(xiě)的話就是默認(rèn)值,上面也寫(xiě)了,默認(rèn)值就是1500毫秒。
@SingleClick(2000)
public void btnAop(View view) {
ToastUtils.showShort("btnAop");
Log.e(TAG, "btnAop");
}
“是不是很簡(jiǎn)單???”
“大哥,我也不想寫(xiě)這一大堆,我只想用,你能封裝成一個(gè)庫(kù)嗎?我用的時(shí)候直接調(diào)用就行!”
“哎,行吧,我封裝一下。。。。”
封裝
“我已經(jīng)將上面第四種方式封裝為了一個(gè)庫(kù)并上傳到了JitPack庫(kù),下面給你說(shuō)一下使用方法吧!”
使用方法很簡(jiǎn)單,需要幾步配置,配置完成之后直接添加注解即可使用,下面是配置方法: 1、在項(xiàng)目的build.gradle中的buildscript中的dependencies添加:
dependencies {
...
classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.4'
}
2、在項(xiàng)目的build.gradle中的allprojects中的repositories添加:
allprojects {
repositories {
...
maven { url 'https://jitpack.io' }
}
}
3、在app的build.gradle中的最上面添加
apply plugin: 'android-aspectjx'
4、在app的build.gradle中的dependencies添加
implementation 'com.github.zhujiang521:AndroidAOP:1.0.1'
5、在你需要使用的方法上面添加上注解即可:
@SingleClick(2000)
public void btnAop(View view) {
ToastUtils.showShort("btnAop");
Log.e(TAG, "btnAop");
}
“小子,會(huì)用了嗎?”
”對(duì)了大哥,我的項(xiàng)目中用的是Kotlin啊,我看你寫(xiě)的都是Java,我那里面能用嘛!”
“吆喝,還Kotlin呢,放心吧,一樣使用!”
@SingleClick
override fun onClick(v: View?) {
if (v != null) {
when(v.id){
R.id.btnClick ->{
ToastUtils.showShort("哈哈哈")
Log.e("哈哈哈","wwww")
}
}
}
}
總結(jié)
“這回可以了吧?”
“可以了大哥,嘿嘿,感謝大哥??”
“小子,快去改項(xiàng)目吧!”
最后放一下項(xiàng)目的地址:https://github.com/zhujiang521/AndroidAOP
以上就是Android開(kāi)發(fā)一行代碼解決安卓重復(fù)點(diǎn)擊的詳細(xì)內(nèi)容,更多關(guān)于Android開(kāi)發(fā)重復(fù)點(diǎn)擊的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android 驅(qū)動(dòng)編寫(xiě)LED-NDK程序
這篇文章主要介紹了Android 驅(qū)動(dòng)編寫(xiě)LED-NDK程序的相關(guān)資料,需要的朋友可以參考下2016-09-09
Android自定義View實(shí)現(xiàn)簡(jiǎn)單水波紋效果
這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)簡(jiǎn)單水波紋效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08
Android編程實(shí)現(xiàn)webview將網(wǎng)頁(yè)打包成apk的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)webview將網(wǎng)頁(yè)打包成apk的方法,以打包HTML5為例分析了webview打包apk的相關(guān)方法、屬性與事件操作技巧,需要的朋友可以參考下2017-08-08
python只需30行代碼就能記錄鍵盤(pán)的一舉一動(dòng)
這篇文章主要介紹了如何用python只寫(xiě)30行代碼就能記錄鍵盤(pán)的一舉一動(dòng),感興趣的同學(xué)快來(lái)看看吧,新手小白也能掌握2021-08-08
Android實(shí)現(xiàn)橫豎屏切換的實(shí)例代碼
本篇文章主要介紹了Android實(shí)現(xiàn)橫豎屏切換的實(shí)例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-06-06
Android 8.1 Launcher3實(shí)現(xiàn)動(dòng)態(tài)指針時(shí)鐘功能
這篇文章主要介紹了Android 8.1 Launcher3實(shí)現(xiàn)動(dòng)態(tài)指針時(shí)鐘功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-07-07
詳解Android 基于TCP和UDP協(xié)議的Socket通信
這篇文章主要介紹了詳解Android 基于TCP和UDP協(xié)議的Socket通信,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11

