ahooks 是怎麼解決用戶多次提交問題?

来源:https://www.cnblogs.com/gopal/archive/2022/08/12/16577831.html
-Advertisement-
Play Games

本文是深入淺出 ahooks 源碼系列文章的第四篇,該系列已整理成文檔-地址。覺得還不錯,給個 star 支持一下哈,Thanks。 本文來探索一下 ahooks 的 useLockFn。並由此討論一個很常見的場景,取消重覆請求。 場景 試想一下,有這麼一個場景,有一個表單,你可能多次提交,就很可能 ...


本文是深入淺出 ahooks 源碼系列文章的第四篇,該系列已整理成文檔-地址。覺得還不錯,給個 star 支持一下哈,Thanks。

本文來探索一下 ahooks 的 useLockFn。並由此討論一個很常見的場景,取消重覆請求。

場景

試想一下,有這麼一個場景,有一個表單,你可能多次提交,就很可能導致結果不正確。

解決這類問題的方法有很多,比如添加 loading,在第一次點擊之後就無法再次點擊。另外一種方法就是給請求非同步函數添加上一個靜態鎖,防止併發產生。這就是 ahooks 的 useLockFn 做的事情。

useLockFn

useLockFn 用於給一個非同步函數增加競態鎖,防止併發執行。

它的源碼比較簡單,如下所示:

import { useRef, useCallback } from 'react';

// 用於給一個非同步函數增加競態鎖,防止併發執行。
function useLockFn<P extends any[] = any[], V extends any = any>(fn: (...args: P) => Promise<V>) {
  // 是否現在處於一個鎖中
  const lockRef = useRef(false);
  // 返回的是增加了競態鎖的函數
  return useCallback(
    async (...args: P) => {
      // 判斷請求是否正在進行
      if (lockRef.current) return;
      // 請求中
      lockRef.current = true;
      try {
        // 執行原有請求
        const ret = await fn(...args);
        // 請求完成,狀態鎖設置為 false
        lockRef.current = false;
        return ret;
      } catch (e) {
        // 請求失敗,狀態鎖設置為 false
        lockRef.current = false;
        throw e;
      }
    },
    [fn],
  );
}

export default useLockFn;

可以看到,它的入參是非同步函數,返回的是一個增加了競態鎖的函數。通過 lockRef 做一個標識位,初始化的時候它的值為 false。當正在請求,則設置為 true,從而下次再調用這個函數的時候,就直接 return,不執行原函數,從而達到加鎖的目的。

缺點

雖然實用,但缺點很明顯,我需要給每一個需要添加競態鎖的請求非同步函數都手動加一遍。那有沒有比較通用和方便的方法呢?

答案是可以通過 axios 自動取消重覆請求。

axios 自動取消重覆請求

axios 取消請求

對於原生的 XMLHttpRequest 對象發起的 HTTP 請求,可以調用 XMLHttpRequest 對象的 abort 方法。

那麼我們項目中常用的 axios 呢?它其實底層也是用的 XMLHttpRequest 對象,它對外暴露取消請求的 API 是 CancelToken。可以使用如下:

const CancelToken = axios.CancelToken;
const source = CancelToken.source();

axios.post('/user/12345', {
  name: 'gopal'
}, {
  cancelToken: source.token
})

source.cancel('Operation canceled by the user.'); // 取消請求,參數是可選的

另外一種使用的方法是調用 CancelToken 的構造函數來創建 CancelToken,具體使用如下:

const CancelToken = axios.CancelToken;
let cancel;

axios.get('/user/12345', {
  cancelToken: new CancelToken(function executor(c) {
    cancel = c;
  })
});

cancel(); // 取消請求

如何自動取消重覆的請求

知道瞭如何取消請求,那怎麼做到自動取消呢?答案是通過 axios 的攔截器。

  • 請求攔截器:該類攔截器的作用是在請求發送前統一執行某些操作,比如在請求頭中添加 token 相關的欄位。
  • 響應攔截器:該類攔截器的作用是在接收到伺服器響應後統一執行某些操作,比如發現響應狀態碼為 401 時,自動跳轉到登錄頁。

具體的做法如下:

第一步,定義幾個重要的輔助函數。

  • generateReqKey:用於根據當前請求的信息,生成請求 Key。只有 key 相同才會判定為是重覆請求。這一點很重要,而且可能跟具體的業務場景有關,比如有一種請求,輸入框模糊搜索,用戶高頻輸入關鍵字,一次性發出多個請求,可能先發出的請求,最後才響應,導致實際搜索結果與預期不符。這種其實就只需要根據 URL 和請求方法判定其為重覆請求,然後取消之前的請求就可以了。

這裡我認為,如果有需要的話,可以暴露一個 API 給開發者進行自定義重覆的規則。這裡我們先根據請求方法、url、以及參數生成唯一的 key 去做。

function generateReqKey(config) {
  const { method, url, params, data } = config;
  return [method, url, Qs.stringify(params), Qs.stringify(data)].join("&");
}
  • addPendingRequest。用於把當前請求信息添加到 pendingRequest 對象中。
const pendingRequest = new Map();
function addPendingRequest(config) {
  const requestKey = generateReqKey(config);
  config.cancelToken = config.cancelToken || new axios.CancelToken((cancel) => {
    if (!pendingRequest.has(requestKey)) {
       pendingRequest.set(requestKey, cancel);
    }
  });
}
  • removePendingRequest。檢查是否存在重覆請求,若存在則取消已發的請求
function removePendingRequest(config) {
  const requestKey = generateReqKey(config);
  if (pendingRequest.has(requestKey)) {
     const cancelToken = pendingRequest.get(requestKey);
     cancelToken(requestKey);
     pendingRequest.delete(requestKey);
  }
}

第二步,添加請求攔截器。

axios.interceptors.request.use(
  function (config) {
    removePendingRequest(config); // 檢查是否存在重覆請求,若存在則取消已發的請求
    addPendingRequest(config); // 把當前請求信息添加到pendingRequest對象中
    return config;
  },
  (error) => {
     return Promise.reject(error);
  }
);

第二步,添加響應攔截器。

axios.interceptors.response.use(
  (response) => {
     removePendingRequest(response.config); // 從pendingRequest對象中移除請求
     return response;
   },
   (error) => {
      removePendingRequest(error.config || {}); // 從pendingRequest對象中移除請求
      if (axios.isCancel(error)) {
        console.log("已取消的重覆請求:" + error.message);
      } else {
        // 添加異常處理
      }
      return Promise.reject(error);
   }
);

到這一步,我們就通過 axios 完成了自動取消重覆請求的功能。

思考與總結

雖然可以通過類似 useLockFn 這樣的 hook或方法給請求函數添加競態鎖的方式解決重覆請求的問題。但這種還是需要依賴於開發者的習慣,如果沒有一些規則的約束,很難避免問題。

通過 axios 攔截器以及其 CancelToken 功能,我們能夠在攔截器中自動將已發的請求取消,當然假如有一些介面就是需要重覆發送請求,可以考慮加一下白名單功能,讓請求不進行取消。

參考


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2022年8月9日下午,StoneDB資料庫主體研發單位石原子科技與華為、openGauss開源社區、雲和恩墨、甲骨文等組織受邀參與《開源資料庫服務商服務能力分級要求》標準第一次討論會。 重點討論:標準的能力域與能力項的分類。 後續,項目組將會繼續完善能力分級要求,推進開源資料庫標準的編製工作。至今 ...
  • 證券行業屬於數據密集型行業,海量高價值歷史數據和爆髮式用戶增長帶來的海量實時數據是證券行業的發展引擎。這也就意味著證券行業對資料庫提出了更高要求,曾經的國外傳統集中式資料庫主流解決方案弊端逐漸暴露。國產資料庫市場亟待完善,騰訊雲資料庫作為國產化建設中的重要一員,為廣大客戶提供滿意的國產化解決方案。最 ...
  • 時代在召喚: HTAP Is On The Way 近些年,HTAP 正在受到人們越來越多的關註,Gartner 在 2014 年提出了 HTAP 這個術語和它的定義: Hybrid transaction/analytical processing (HTAP) is an emerging ap ...
  • 序: 記憶體關係資料庫沒有找到開源好用的,很多都是商用。雖然mysql有memory引擎,但寫是整體鎖表,沒法用。 一直想將mysql放入記憶體中,搜索n次資料,沒找到合適的,可能之前思路不對。 最近在測試將mysql放在linux的tmpfs記憶體盤中,網上找了不少方法,都啟動失敗。突然想到直接掛載方式 ...
  • 近日,一體化實時 HTAP 資料庫 StoneDB 與華為鯤鵬 920 已完成並通過相互相容性測試認證,取得 KUNPENG COMPATIBLE 證書及認證徽標的使用權。 該認證意味著 StoneDB 在相容性、穩定性、性能、安全、功耗、功能等六大維度的測試均符合鯤鵬技術標準,可以適配華為鯤鵬 9 ...
  • 原文:Android自定義View學習(1)——基礎知識介紹 - Stars-One的雜貨小窩 準備學習自定義View,介紹一下先瞭解了下相關的前置基礎知識,特此總結 本系列集合文章鏈接可訪問Android自定義View學習系列教程 坐標系 屏幕坐標系 不管是Android設備,還是PC設備,屏幕坐 ...
  • 圖像中的文字內容通常包含了很重要的信息,然而由於拍攝限制、解析度過低、拍攝對象過遠等等原因,文字內容可能模糊不清。比如在拍照上傳發票時,因為聚焦不准、燈光等問題,導致App無法自動識別發票內容;一些檔案要上傳到後臺資料庫里時,由於年代久遠,有些文字已無法辨認;被保存轉發多次的截圖,由於每個應用壓縮程 ...
  • 本文是深入淺出 ahooks 源碼系列文章的第五篇,該系列已整理成文檔-地址。覺得還不錯,給個 star 支持一下哈,Thanks。 本文來探索一下 ahooks 是怎麼封裝 React 的一些執行“時機”的? Function Component VS Class Component 學習類似 R ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...