## 防抖技術
防抖(Debouncing)是一種編程技術,用于控制事件處理函數的執行頻率。在用戶與界面交互頻繁的場景中,比如連續滾動、連續輸入等,如果每次交互都觸發事件處理函數,可能會導致性能問題或不必要的數據庫操作。
防抖技術通過設定一個延遲時間,在這段時間內,即使觸發了多次事件,事件處理函數也只會在延遲時間結束后執行一次。如果在這個延遲時間內再次觸發事件,那么之前的延遲會被重置,重新開始計算延遲時間。這樣,只有最后一次事件觸發后,延遲時間結束后,事件處理函數才會執行。
> 防抖技術常用于以下場景
- 搜索框輸入:用戶連續輸入時,只有輸入停止一段時間后才觸發搜索請求。
- 窗口調整大小:用戶調整窗口大小時,只有調整結束后才執行相關操作。
- 滾動事件:用戶滾動頁面時,只有滾動停止一段時間后才進行數據處理。
### 解決方案
在Web系統的交互設計中,表單提交是一個核心功能,但若不加以適當控制,用戶誤操作或網絡的不穩定性都可能導致同一請求被重復發送,從而產生冗余數據。為了應對這一挑戰,我們可以從兩個層面進行優化:
1. **前端防抖**:通過在用戶界面上實現按鈕的加載狀態(loading state),可以有效防止用戶因手抖而重復點擊,從而避免前端生成多個請求。
2. **后端防抖**:對于由網絡波動引起的請求重發問題,前端的控制措施顯得不夠充分。因此,后端需要引入防抖邏輯,通過識別請求的唯一性(例如使用請求ID或時間戳),確保即便在網絡不穩定的情況下,同一請求也不會被重復處理。
防抖策略是確保Web系統穩定性和數據一致性的關鍵。前端的防抖措施提升了用戶體驗,而后端的防抖措施則保障了數據的準確性和系統的健壯性。兩者結合,可以構建一個更加穩定和用戶友好的Web應用環境。通過這種雙重保障,我們可以有效地減少因誤操作或網絡問題導致的重復請求,維護系統的高效運行。
## 防抖場景
在Web系統中,并非所有接口都需要防抖,但以下類型的接口通常可以從防抖機制中獲益:
#### 表單輸入場景
* **搜索框輸入**:用戶在搜索框中輸入時,可能會觸發實時搜索或自動完成功能。防抖可以減少因快速輸入導致的頻繁請求。
* **表單輸入**:尤其是那些包含多個字段或需要進行復雜驗證的表單,防抖可以避免用戶因誤操作而重復提交。
#### 按鈕點擊場景
按鈕點擊類接口,如提交表單或保存設置,用戶在操作過程中可能會因各種原因頻繁點擊按鈕,這不僅可能影響用戶體驗,還可能導致不必要的服務器請求,增加系統負擔。
為了防止用戶因急促操作而導致的頻繁請求。通過設置一個短暫的等待時間,只有在用戶停止點擊達到預設的時間閾值后,才會觸發實際的請求發送。這種方法不僅減少了服務器的負擔,也避免了因重復請求而可能產生的數據錯誤或沖突。
#### 滾動加載場景
在滾動加載類接口中,如下拉刷新、上拉加載等,用戶的操作往往伴隨著連續的滾動動作。為了提升系統效率并避免因頻繁觸發而導致的性能問題。通過設定一個合理的時間間隔,只有在用戶滾動動作停止一段時間后,系統才會執行請求發送,從而實現智能的請求管理。
## 如何防抖
### 使用共享緩存

*圖片來源:https://developer.aliyun.com/article/1541251*
### 使用分布式鎖

*圖片來源:https://developer.aliyun.com/article/1541251*
常見的分布式組件有Redis、Zookeeper等,但結合實際業務來看,一般都會選擇Redis,因為Redis一般都是Web系統必備的組件,不需要額外搭建。
## 代碼實現
模仿一個用戶添加接口
目前數據庫表中沒有對`mobile`字段做UK唯一索引限制,這就會導致每調用一次`userAdd`就會創建一個用戶,即使`mobile`相同。
`demo_user` 表結構
```sql
CREATE TABLE `demo_user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`username` varchar(32) NOT NULL,
`mobile` char(13) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
```
### 分布式鎖
RedisLock.php
```
<?php
/**
* @desc RedisLock.php 描述信息
* @author Tinywan(ShaoBo Wan)
* @date 2024/6/23 8:31
*/
declare(strict_types=1);
namespace app\common\service;
use support\Redis;
class RedisLock
{
// 分布式并發鎖
const DISTRIBUTED_CONCURRENT_LOCK = 'DISTRIBUTED_CONCURRENT_LOCK:';
/**
* @desc: 獲取鎖
* @param string $lock_name
* @param int $acquire_time
* @param int $lock_timeout
* @return bool|string
* @author Tinywan(ShaoBo Wan)
*/
public static function getLockWithTimeout(string $lock_name, int $acquire_time = 3, int $lock_timeout = 20)
{
$identifier = md5($_SERVER['REQUEST_TIME'] . mt_rand(1, 10000000));
$lock_name = self::DISTRIBUTED_CONCURRENT_LOCK . $lock_name;
$lock_timeout = intval(ceil($lock_timeout));
$end_time = time() + $acquire_time;
while (time() < $end_time) {
$script = <<<luascript
local result = redis.call('setnx',KEYS[1],ARGV[1]);
if result == 1 then
return redis.call('expire',KEYS[1],ARGV[2])
elseif redis.call('ttl',KEYS[1]) == -1 then
return redis.call('expire',KEYS[1],ARGV[2]) -- 續租(renew)
else
return 0
end
luascript;
$result = Redis::eval($script, 1, $lock_name, $identifier, $lock_timeout);
if ($result === 1) {
return $identifier;
}
usleep(100000);
}
return false;
}
/**
* @desc: 釋放鎖
* @param string $lock_name
* @param string $identifier
* @return bool
* @author Tinywan(ShaoBo Wan)
*/
public static function releaseLock(string $lock_name, string $identifier): bool
{
$lock_name = self::DISTRIBUTED_CONCURRENT_LOCK . $lock_name;
while (true) {
$script = <<<luascript
if redis.call("get",KEYS[1]) == ARGV[1] then
return redis.call("del",KEYS[1]);
else
return 0
end
luascript;
$result = Redis::eval($script, 1, $lock_name, $identifier);
if ($result == 1) {
return true;
}
break;
}
return false;
}
}
```
### 業務代碼實現
```
<?php
/**
* @desc Demo.php 描述信息
* @author Tinywan(ShaoBo Wan)
* @date 2024/6/23 20:14
*/
declare(strict_types=1);
namespace app\controller;
use app\common\service\RedisLock;
use support\Request;
use support\Response;
use Tinywan\ExceptionHandler\Exception\BadRequestHttpException;
use Tinywan\ExceptionHandler\Exception\ServerErrorHttpException;
class DemoController
{
/**
* @desc 用戶添加
* @param Request $request
* @return Response
* @throws BadRequestHttpException
* @throws ServerErrorHttpException
* @author Tinywan(ShaoBo Wan)
*/
public function userAdd(Request $request): Response
{
$param = $request->post();
/** 鎖名稱 */
$lockName = (string) $param['mobile'];
/** 嘗試獲取搶占鎖標識 */
$lockIdentifier = RedisLock::getLockWithTimeout($lockName);
/** 沒有拿到鎖說明已經有了請求了 */
if (false === $lockIdentifier) {
throw new BadRequestHttpException('您的操作太快啦!請不要連續點擊提交');
}
try {
/** 進行業務處理 */
\think\facade\Db::table('demo_user')->insert($param);
/** 進行業務處理 */
} catch (\Throwable $throwable) {
/** 釋放鎖 */
RedisLock::releaseLock($lockName, $lockIdentifier);
throw new ServerErrorHttpException('系統異常:' . $throwable->getMessage());
}
/** 釋放鎖 */
RedisLock::releaseLock($lockName, $lockIdentifier);
return json(['code' => 200, 'msg' => 'success']);
}
}
```
RedisLock的核心思路就是搶鎖,當一次請求搶到鎖之后,對鎖加一個過期時間,在這個時間段內重復的請求是無法獲得這個鎖。
### 驗證分布式鎖
> 正確提交

> 后端異常提交

> 后端未響應之前提交

> 相同時間段內重復,鎖釋放剩余時間

- 設計模式系列
- 工廠方法模式
- 序言
- Windows程序注冊為服務的工具WinSW
- 基礎
- 安裝
- 開發規范
- 目錄結構
- 配置
- 快速入門
- 架構
- 請求流程
- 架構總覽
- URL訪問
- 容器和依賴注入
- 中間件
- 事件
- 代碼層結構
- 四個層次
- 路由
- 控制器
- 請求
- 響應
- 數據庫
- MySQL實時同步數據到ES解決方案
- 阿里云DTS數據MySQL同步至Elasticsearch實戰
- PHP中的MySQL連接池
- PHP異步非阻塞MySQL客戶端連接池
- 模型
- 視圖
- 注解
- @SpringBootApplication(exclude={DataSourceAutoConfiguration.calss})
- @EnableFeignClients(basePackages = "com.wotu.feign")
- @EnableAspectJAutoProxy
- @EnableDiscoveryClient
- 錯誤和日志
- 異常處理
- 日志處理
- 調試
- 驗證
- 驗證器
- 驗證規則
- 擴展庫
- 附錄
- Spring框架知識體系詳解
- Maven
- Maven和Composer
- 構建Maven項目
- 實操課程
- 01.初識SpringBoot
- 第1章 Java Web發展史與學習Java的方法
- 第2章 環境與常見問題踩坑
- 第3章 springboot的路由與控制器
- 02.Java編程思想深度理論知識
- 第1章 Java編程思想總體
- 第2章 英雄聯盟的小案例理解Java中最為抽象的概念
- 第3章 徹底理解IOC、DI與DIP
- 03.Spring與SpringBoot理論篇
- 第1章 Spring與SpringBoot導學
- 第2章 Spring IOC的核心機制:實例化與注入
- 第3章 SpringBoot基本配置原理
- 04.SprinBoot的條件注解與配置
- 第1章 conditonal 條件注解
- 第2章 SpringBoot自動裝配解析
- 05.Java異常深度剖析
- 第1章 Java異常分類剖析與自定義異常
- 第2章 自動配置Url前綴
- 06.參數校驗機制與LomBok工具集的使用
- 第1章 LomBok工具集的使用
- 第2章 參數校驗機制以及自定義校驗
- 07.項目分層設計與JPA技術
- 第1章 項目分層原則與層與層的松耦合原則
- 第2章 數據庫設計、實體關系與查詢方案探討
- 第3章 JPA的關聯關系與規則查詢
- 08.ORM的概念與思維
- 第1章 ORM的概念與思維
- 第2章 Banner等相關業務
- 第3章 再談數據庫設計技巧與VO層對象的技巧
- 09.JPA的多種查詢規則
- 第1章 DozerBeanMapper的使用
- 第2章 詳解SKU的規格設計
- 第3章 通用泛型Converter
- 10.令牌與權限
- 第1章 通用泛型類與java泛型的思考
- 常見問題
- 微服務
- demo
- PHP中Self、Static和parent的區別
- Swoole-Cli
- 為什么要使用現代化PHP框架?
- 公眾號
- 一鍵部署微信公眾號Markdown編輯器(支持適配和主題設計)
- Autodesigner 2.0發布
- Luya 一個現代化PHP開發框架
- PHPZip - 創建、讀取和管理 ZIP 文件的簡單庫
- 吊打Golang的PHP界天花板webman壓測對比
- 簡潔而強大的 YAML 解析庫
- 推薦一個革命性的PHP測試框架:Kahlan
- ServBay下一代Web開發環境
- 基于Websocket和Canvas實現多人協作實時共享白板
- Apipost預執行腳本如何調用外部PHP語言
- 認證和授權的安全令牌 Bearer Token
- Laradock PHP 的 Docker 完整本地開發環境
- 高效接口防抖策略,確保數據安全,避免重復提交的終極解決方案!
- TIOBE 6月榜單:PHP穩步前行,編程語言生態的微妙變化
- Aho-Corasick字符串匹配算法的實現
- Redis鍵空間通知 Keyspace Notification 事件訂閱
- ServBay如何啟用并運行Webman項目
- 使用mpdf實現導出pdf文件功能
- Medoo 輕量級PHP數據庫框架
- 在PHP中編寫和運行單元測試
- 9 PHP運行時基準性能測試
- QR碼生成器在PHP中的源代碼
- 使用Gogs極易搭建的自助Git服務
- Gitea
- webman如何記錄SQL到日志?
- Sentry PHP: 實時監測并處理PHP應用程序中的錯誤
- Swoole v6 Alpha 版本已發布
- Proxypin
- Rust實現的Redis內存數據庫發布
- PHP 8.4.0 Alpha 1 測試版本發布
- 121
- Golang + Vue 開發的開源輕量 Linux 服務器運維管理面板
- 內網穿透 FRP VS Tailscale
- 新一代開源代碼托管平臺Gitea
- 微服務系列
- Nacos云原生配置中心介紹與使用
- 輕量級的開源高性能事件庫libevent
- 國密算法
- 國密算法(商用密碼)
- GmSSL 支持國密SM2/SM3/SM4/SM9/SSL 密碼工具箱
- GmSSL PHP 使用
- 數據庫
- SQLite數據庫的Web管理工具
- 阿里巴巴MySQL數據庫強制規范
- PHP
- PHP安全測試秘密武器 PHPGGC
- 使用declare(strict_types=1)來獲得更健壯的PHP代碼
- PHP中的魔術常量
- OSS 直傳阿里騰訊示例
- PHP源碼編譯安裝APCu擴展實現數據緩存
- BI性能DuckDB數據管理系統
- 為什么別人可以是架構師!而我卻不是?
- 密碼還在用 MD5 加鹽?不如試試 password_hash
- Elasticsearch 在電商領域的應用與實踐
- Cron 定時任務入門
- 如何動態設置定時任務!而不是寫死在Linux Crontab
- Elasticsearch的四種查詢方式,你知道多少?
- Meilisearch vs Elasticsearch
- OpenSearch vs Elasticsearch
- Emlog 輕量級開源博客及建站系統
- 現代化PHP原生協程引擎 PRipple
- 使用Zephir編寫C擴展將PHP源代碼編譯加密
- 如何將PHP源代碼編譯加密,同時保證代碼能正常的運行
- 為什么選擇Zephir給PHP編寫動態擴展庫?
- 使用 PHP + XlsWriter實現百萬級數據導入導出
- Rust編寫PHP擴展
- 阿里云盤開放平臺對接進行文件同步
- 如何構建自己的PHP靜態可執行文件
- IM后端架構
- RESTful設計方法和規范
- PHP編譯器BPC 7.3 發布,成功編譯ThinkPHP8
- 高性能的配置管理擴展 Yaconf
- PHP實現雪花算法庫 Snowflake
- PHP官方現代化核心加密庫Sodium
- pie
- 現代化、精簡、非阻塞PHP標準庫PSL
- PHP泛型和集合
- 手把手教你正確使用 Composer包管理
- JWT雙令牌認證實現無感Token自動續期
- 最先進PHP大模型深度學習庫TransformersPHP
- PHP如何啟用 FFI 擴展
- PHP超集語言PXP
- 低延遲雙向實時事件通信 Socket.IO
- PHP OOP中的繼承和多態
- 強大的現代PHP高級調試工具Kint
- PHP基金會
- 基于webman+vue3高質量中后臺框架SaiAdmin
- 開源免費的定時任務管理系統:Gocron
- 簡單強大OCR工具EasyOCR在PHP中使用
- PHP代碼抽象語法樹工具PHP AST Viewer
- MySQL數據庫管理工具PHPMyAdmin
- Rust編寫的一款高性能多人代碼編輯器Zed
- 超高性能PHP框架Workerman v5.0.0-beta.8 發布
- 高并發系列
- 入門介紹及安裝
- Lua腳本開發 Hello World
- 執行流程與階段詳解
- Nginx Lua API 接口開發
- Lua模塊開發
- OpenResty 高性能的正式原因
- 記一次查找 lua-resty-mysql 庫 insert_id 的 bug
- 包管理工具OPM和LuaRocks使用
- 異步非阻塞HTTP客戶端庫 lua-resty-http
- Nginx 內置綁定變量
- Redis協程網絡庫 lua-resty-redis
- 動態HTML渲染庫 lua-testy-template
- 單獨的
- StackBlitz在線開發環境
- AI
- 基礎概念
- 12312
- 基礎鏡像的坑
- 利用phpy實現 PHP 編寫 Vision Transformer (ViT) 模型
- 語義化版本 2.0.0