Android 按鈕快速點擊問題的解決方案!


快速點擊可能帶來的問題:

1.同一個接口調用多次

2.相同的activity啓動多個


針對上面2個問題,參照了2篇文章,給出我認爲比較好的解決方案。


第一個問題:同一個接口調用多次的問題

參考:https://www.jianshu.com/p/06c5b35b4e51

 mButton.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

                if (AntiShakeUtils.isInvalidClick(v)) {
                    return;
                }
                 //TODO 你要調用的接口或者其他事情
            }
        });

AntiShakeUtils 類:


import android.support.annotation.IntRange;
import android.support.annotation.NonNull;
import android.view.View;

import com.vwiov.vwiovdriverapp.R;


/**
 * 防抖動點擊
 * <br>Email:[email protected]
 * <br>QQ:1006368252
 * <br><a href="https://github.com/JustinRoom/JSCKit" target="_blank">https://github.com/JustinRoom/JSCKit</a>
 *
 * @author jiangshicheng
 */
public class AntiShakeUtils {

    private final static long INTERNAL_TIME = 1000;

    /**
     * Whether this click event is invalid.
     *
     * @param target target view
     * @return true, invalid click event.
     * @see #isInvalidClick(View, long)
     */
    public static boolean isInvalidClick(@NonNull View target) {
        return isInvalidClick(target, INTERNAL_TIME);
    }

    /**
     * Whether this click event is invalid.
     *
     * @param target       target view
     * @param internalTime the internal time. The unit is millisecond.
     * @return true, invalid click event.
     */
    public static boolean isInvalidClick(@NonNull View target, @IntRange(from = 0) long internalTime) {
        long curTimeStamp = System.currentTimeMillis();
        long lastClickTimeStamp = 0;
        Object o = target.getTag(R.id.last_click_time);
        if (o == null){
            target.setTag(R.id.last_click_time, curTimeStamp);
            return false;
        }
        lastClickTimeStamp = (Long) o;
        boolean isInvalid = curTimeStamp - lastClickTimeStamp < internalTime;
        if (!isInvalid)
            target.setTag(R.id.last_click_time, curTimeStamp);
        return isInvalid;
    }
}

第二個問題:相同的activity啓動多個的問題

參考:https://blog.csdn.net/ziyexiaoxiao/article/details/85292729

將下面的內容在BaseActivity裏定義就好。

    /**
     * 每次啓動activity都會調用此方法
     */
    @SuppressLint("RestrictedApi")
    @Override
    public void startActivityForResult(Intent intent, int requestCode, @Nullable Bundle options) {
        if (checkDoubleClick(intent)) {
            super.startActivityForResult(intent, requestCode, options);
 
        }
    }
    private String mActivityJumpTag;        //activity跳轉tag
    private long mClickTime;                //activity跳轉時間
 
    /**
     * 檢查是否重複跳轉,不需要則重寫方法並返回true
     */
    protected boolean checkDoubleClick(Intent intent) {
 
        // 默認檢查通過
        boolean result = true;
        // 標記對象
        String tag;
        if (intent.getComponent() != null) { // 顯式跳轉
            tag = intent.getComponent().getClassName();
        }else if (intent.getAction() != null) { // 隱式跳轉
            tag = intent.getAction();
        }else {
            return true;
        }
 
        if (tag.equals(mActivityJumpTag) && mClickTime >= SystemClock.uptimeMillis() - 500) {
            // 檢查不通過
            result = false;
        }
 
        // 記錄啓動標記和時間
        mActivityJumpTag = tag;
        mClickTime = SystemClock.uptimeMillis();
        return result;
    }

 針對fragment 點擊跳轉問題,只需要在承載它的activity調用上面的方法就可以的,當然集成BaseActivity的不需要處理!


如果碰到個別activity還是會啓動相同,別的activity不會的情況。

排查下對應activity的onCreate,onStart,onResume是否進行了耗時操作,耗時操作異步處理!

原因:

A跳轉到B,需要經歷A的onPause,B的onCreate,onStart,onResume,所以在這幾個方法中不能做耗時的操作。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章