## redis 實現鎖
```php
// 其他服務 不在注冊 register_shutdown_function 而是添加 事件
$shutdownEvent = [];
// 注冊一個會在php中止時執行的函數
register_shutdown_function('shutdown_function');
function shutdown_function()
{
global $shutdownEvent;
// 執行注冊的事件
foreach ($shutdownEvent as $key => $f) {
if (is_callable($f)) {
$f();
}
}
$e = error_get_last();
if ($e) {
$errorInfo = friendlyErrorType($e['type']);
// print_r($e);
switch ($e['type']) {
case E_PARSE:
case E_ERROR:
case E_CORE_ERROR:
case E_COMPILE_ERROR:
case E_USER_ERROR:
$errorStr = "appErrorHandler: type: {$e['type']} ({$errorInfo[0]} - {$errorInfo[1]}); message: {$e['message']}; file: {$e['file']}; line: {$e['line']}";
trace($errorStr, 'run_error');
trace('[shutdown_function:trace]', '', true);
error(1, $errorStr);
break;
}
}
// 寫入日志
trace('[shutdown_function:trace]', '', true);
}
// 使用 redis實現的簡單無阻塞的鎖(沒有考慮復雜的情況)
// https://www.cnblogs.com/fengff/p/10913492.html
function lock($key, $ok, $gone)
{
global $redis, $shutdownEvent;
$roomid = 'room_' . time() . rand(1, 9999);
// 初始化一個空函數防止下面報錯,任何環節,代碼節點都有可能發生異常甚至不確定性故障
$delLock = function () {
};
// trace('lock: ' . $key . PHP_EOL, '>debug-lock');
// exit;
try {
$isLockOk = $redis->setNX($key, $roomid);
// var_dump($isLockOk);exit;
if ($isLockOk) {
// 獲得鎖
// echo PHP_EOL . 'lock ok!' . PHP_EOL . PHP_EOL;
// 防止A進程獲得鎖后,由于A執行時間太長,鎖被回收了
// 此時B獲得鎖,等A執行完畢,刪除時刪除了B持有鎖,它以為刪除的是自己的,因為它并不是到自己的鎖被回收了
// 設置過期是因為防止任務掛了鎖沒被回收,這個問題不會有了,所以我們不設置過期
// 不能因為 執行時間長的 設置過期,不然還會有兩個進行并發了
// 但是為了規范嚴謹,我們刪除時還是嚴格檢查
$delLock = function () use ($key, $redis, $roomid)
{
if ($redis->get($key) == $roomid) {
$redis->del($key);
trace('delLock: ' . $key . ' ' . $roomid . PHP_EOL, '>debug-lock');
// echo 'delLock: ' . $key . ' ' . $roomid . PHP_EOL;
}
};
// trace('lock: ok!', '>debug-lock');
// 能保存上下文
// 像js的閉包一樣,贊!
$shutdownEvent['unLock'] = function () use ($key, $redis, $roomid, $delLock) {
// 只有發生錯誤時,才使用事件刪除key
// ctrl + c 退出就沒辦法了(不算異常),除非能獲取腳本推出code
if (!error_get_last()) {
return;
}
// 獲取了鎖的腳本,腳本退出時 刪也沒事
// ctrl + c 退出 register_shutdown_function 不會執行
// 這種情況那就只有使用信號了
trace('shutdownEvent unLock: ' . $key . ' ' . $roomid . PHP_EOL, '>debug-lock');
// echo 'shutdownEvent unLock: ' . $key . ' ' . $roomid . PHP_EOL;
// $redis 連接也可能失效,刪除有很多種失敗的情況
// 只能盡量將最容易失敗的地方處理好
$delLock();
};
// 注意: 注冊事件這段代碼必須放在這里: 獲得鎖之后 - 執行代碼之前 , 不可預料的 出錯主要是在 $ok() 這里
// trace('lock: ok -> shutdownEvent', '>debug-lock');
$res = $ok();
// 如果任務發生錯誤崩了,這下面不會執行,但是沒關系 shutdownEvent 事件還會執行
$delLock();
// unset($shutdownEvent['unLock']); // 如果 沒異常,delLock() 順利執行了,就不用繼續執行事件了
} else {
// 當前已在運行,直接退出
// echo PHP_EOL . 'Be gone' . PHP_EOL . PHP_EOL;
$res = $gone();
}
} catch (\Exception $e) {
$delLock();
throw $e;
}
// appErrorHandler: type: 1 (E_ERROR - 致命錯誤); message: Class 'sModel' not found; file: D:\wamp64\www\patent_management_system\include\rockFun.php; line: 564
// todo: 如果出現致命錯誤,會導致 key 沒刪掉
// https://www.php.net/manual/zh/function.register-shutdown-function.php
// 是可以多次注冊的,是追加不是覆蓋,沒有問題
// 只注冊一次
// 這個無法 實現只注冊一次
// register_shutdown_function();
// 改用 注冊 shutdown 事件,框架調用
// 即使這樣肯定也不能萬無一失
return $res;
}
```
----
### 用法:
```php
// 使任務保持“單例”執行
lock('lock_' . $fun, function () use ($fun) {
$fun();
}, function () use ($fun) {
echo "crontab_run: {$fun} task running ..." . PHP_EOL;
// 如果當前任務已經在執行了,就什么都不做
});
```
----
### 待實現
- 可重入
- 阻塞等特性
...
- 開始
- 公益
- 更好的使用看云
- 推薦書單
- 優秀資源整理
- 技術文章寫作規范
- SublimeText - 編碼利器
- PSR-0/PSR-4命名標準
- php的多進程實驗分析
- 高級PHP
- 進程
- 信號
- 事件
- IO模型
- 同步、異步
- socket
- Swoole
- PHP擴展
- Composer
- easyswoole
- php多線程
- 守護程序
- 文件鎖
- s-socket
- aphp
- 隊列&并發
- 隊列
- 講個故事
- 如何最大效率的問題
- 訪問式的web服務(一)
- 訪問式的web服務(二)
- 請求
- 瀏覽器訪問阻塞問題
- Swoole
- 你必須理解的計算機核心概念 - 碼農翻身
- CPU阿甘 - 碼農翻身
- 異步通知,那我要怎么通知你啊?
- 實時操作系統
- 深入實時 Linux
- Redis 實現隊列
- redis與隊列
- 定時-時鐘-阻塞
- 計算機的生命
- 多進程/多線程
- 進程通信
- 拜占庭將軍問題深入探討
- JAVA CAS原理深度分析
- 隊列的思考
- 走進并發的世界
- 鎖
- 事務筆記
- 并發問題帶來的后果
- 為什么說樂觀鎖是安全的
- 內存鎖與內存事務 - 劉小兵2014
- 加鎖還是不加鎖,這是一個問題 - 碼農翻身
- 編程世界的那把鎖 - 碼農翻身
- 如何保證萬無一失
- 傳統事務與柔性事務
- 大白話搞懂什么是同步/異步/阻塞/非阻塞
- redis實現鎖
- 淺談mysql事務
- PHP異常
- php錯誤
- 文件加載
- 路由與偽靜態
- URL模式之分析
- 字符串處理
- 正則表達式
- 數組合并與+
- 文件上傳
- 常用驗證與過濾
- 記錄
- 趣圖
- foreach需要注意的問題
- Discuz!筆記
- 程序設計思維
- 抽象與具體
- 配置
- 關于如何學習的思考
- 編程思維
- 談編程
- 如何安全的修改對象
- 臨時
- 臨時筆記
- 透過問題看本質
- 程序后門
- 邊界檢查
- session
- 安全
- 王垠
- 第三方數據接口
- 驗證碼問題
- 還是少不了虛擬機
- 程序員如何談戀愛
- 程序員為什么要一直改BUG,為什么不能一次性把代碼寫好?
- 碎碎念
- 算法
- 實用代碼
- 相對私密與絕對私密
- 學習目標
- 隨記
- 編程小知識
- foo
- 落盤
- URL編碼的思考
- 字符編碼
- Elasticsearch
- TCP-IP協議
- 碎碎念2
- Grafana
- EFK、ELK
- RPC
- 依賴注入
- 科目一
- 開發筆記
- 經緯度格式轉換
- php時區問題
- 解決本地開發時調用遠程AIP跨域問題
- 后期靜態綁定
- 談tp的跳轉提示頁面
- 無限分類問題
- 生成微縮圖
- MVC名詞
- MVC架構
- 也許模塊不是唯一的答案
- 哈希算法
- 開發后臺
- 軟件設計架構
- mysql表字段設計
- 上傳表如何設計
- 二開心得
- awesomes-tables
- 安全的代碼部署
- 微信開發筆記
- 賬戶授權相關
- 小程序獲取是否關注其公眾號
- 支付相關
- 提交訂單
- 微信支付筆記
- 支付接口筆記
- 支付中心開發
- 下單與支付
- 支付流程設計
- 訂單與支付設計
- 敏感操作驗證
- 排序設計
- 代碼的運行環境
- 搜索關鍵字的顯示處理
- 接口異步更新ip信息
- 圖片處理
- 項目搭建
- 閱讀文檔的新方式
- mysql_insert_id并發問題思考
- 行鎖注意事項
- 細節注意
- 如何處理用戶的輸入
- 不可見的字符
- 抽獎
- 時間處理
- 應用開發實戰
- python 學習記錄
- Scrapy 教程
- Playwright 教程
- stealth.min.js
- Selenium 教程
- requests 教程
- pyautogui 教程
- Flask 教程
- PyInstaller 教程
- 蜘蛛
- python 文檔相似度驗證
- thinkphp5.0數據庫與模型的研究
- workerman進程管理
- workerman網絡分析
- java學習記錄
- docker
- 筆記
- kubernetes
- Kubernetes
- PaddlePaddle
- composer
- oneinstack
- 人工智能 AI
- 京東
- pc_detailpage_wareBusiness
- doc
- 電商網站設計
- iwebshop
- 商品規格分析
- 商品屬性分析
- tpshop
- 商品規格分析
- 商品屬性分析
- 電商表設計
- 設計記錄
- 優惠券
- 生成唯一訂單號
- 購物車技術
- 分類與類型
- 微信登錄與綁定
- 京東到家庫存系統架構設計
- crmeb
- 命名規范
- Nginx https配置
- 關于人工智能
- 從人的思考方式到二叉樹
- 架構
- 今日有感
- 文章保存
- 安全背后: 瀏覽器是如何校驗證書的
- 避不開的分布式事務
- devops自動化運維、部署、測試的最后一公里 —— ApiFox 云時代的接口管理工具
- 找到自己今生要做的事
- 自動化生活
- 開源與漿果
- Apifox: API 接口自動化測試指南