① php 使用redis鎖限制並發訪問類示例
本文介紹了php
使用redis鎖限制並發訪問類,並詳細的介紹了並發訪問限制方法。
1.並發訪問限制問題
對於一些需要限制同一個用戶並發訪問的場景,如果用戶並發請求多次,而伺服器處理沒有加鎖限制,用戶則可以多次請求成功。
例如換領優惠券,如果用戶同一時間並發提交換領碼,在沒有加鎖限制的情況下,用戶則可以使用同一個換領碼同時兌換到多張優惠券。
偽代碼如下:
if
A(可以換領)
B(執行換領)
C(更新為已換領)
D(結束)
如果用戶並發提交換領碼,都能通過可以換領(A)的判斷,因為必須有一個執行換領(B)後,才會更新為已換領(C)。因此如果用戶在有一個更新為已換領之前,有多少次請求,這些請求都可以執行成功。
2.並發訪問限制方法
使用文件鎖可以實現並發訪問限制,但對於分布式架構的環境,使用文件鎖不能保證多台伺服器的並發訪問限制。
Redis是一個開源的使用ANSI
C語言編寫、支持網路、可基於內存亦可持久化的日誌型、Key-Value資料庫,並提供多種語言的API。
本文將使用其setnx方法實現分布式鎖功能。setnx即Set
it
N**ot
eX**ists。
當鍵值不存在時,插入成功(獲取鎖成功),如果鍵值已經存在,則插入失敗(獲取鎖失敗)
RedisLock.class.PHP
<?php
/**
*
Redis鎖操作類
*
Date:
2016-06-30
*
Author:
fdipzone
*
Ver:
1.0
*
*
Func:
*
public
lock
獲取鎖
*
public
unlock
釋放鎖
*
private
connect
連接
*/
class
RedisLock
{
//
class
start
private
$_config;
private
$_redis;
/**
*
初始化
*
@param
Array
$config
redis連接設定
*/
public
function
__construct($config=array()){
$this->_config
=
$config;
$this->_redis
=
$this->connect();
}
/**
*
獲取鎖
*
@param
String
$key
鎖標識
*
@param
Int
$expire
鎖過期時間
*
@return
Boolean
*/
public
function
lock($key,
$expire=5){
$is_lock
=
$this->_redis->setnx($key,
time()+$expire);
//
不能獲取鎖
if(!$is_lock){
//
判斷鎖是否過期
$lock_time
=
$this->_redis->get($key);
//
鎖已過期,刪除鎖,重新獲取
if(time()>$lock_time){
$this->unlock($key);
$is_lock
=
$this->_redis->setnx($key,
time()+$expire);
}
}
return
$is_lock?
true
:
false;
}
/**
*
釋放鎖
*
@param
String
$key
鎖標識
*
@return
Boolean
*/
public
function
unlock($key){
return
$this->_redis->del($key);
}
/**
*
創建redis連接
*
@return
Link
*/
private
function
connect(){
try{
$redis
=
new
Redis();
$redis->connect($this->_config['host'],$this->_config['port'],$this->_config['timeout'],$this->_config['reserved'],$this->_config['retry_interval']);
if(empty($this->_config['auth'])){
$redis->auth($this->_config['auth']);
}
$redis->select($this->_config['index']);
}catch(RedisException
$e){
throw
new
Exception($e->getMessage());
return
false;
}
return
$redis;
}
}
//
class
end
?>
demo.php
<?php
require
'RedisLock.class.php';
$config
=
array(
'host'
=>
'localhost',
'port'
=>
6379,
'index'
=>
0,
'auth'
=>
'',
'timeout'
=>
1,
'reserved'
=>
NULL,
'retry_interval'
=>
100,
);
//
創建redislock對象
$oRedisLock
=
new
RedisLock($config);
//
定義鎖標識
$key
=
'mylock';
//
獲取鎖
$is_lock
=
$oRedisLock->lock($key,
10);
if($is_lock){
echo
'get
lock
success<br>';
echo
'do
sth..<br>';
sleep(5);
echo
'success<br>';
$oRedisLock->unlock($key);
//
獲取鎖失敗
}else{
echo
'request
too
frequently<br>';
}
?>
測試方法:
打開兩個不同的瀏覽器,同時在A,B中訪問demo.php
如果先訪問的會獲取到鎖
輸出
get
lock
success
do
sth..
success
另一個獲取鎖失敗則會輸出request
too
frequently
保證同一時間只有一個訪問有效,有效限制並發訪問。
為了避免系統突然出錯導致死鎖,所以在獲取鎖的時候增加一個過期時間,如果已超過過期時間,即使是鎖定狀態都會釋放鎖,避免死鎖導致的問題。
源碼下載地址:點擊查看
② php 鎖定文件的問題。
問題在於,flock()這個函數不好使。
這是由PHP先天決定的。
用戶A訪問你的程序後,PHP在伺服器端會啟動一個線程,flock這個函數只能在當前線程中鎖住這個文件。
同時,如果有另外的用戶B也訪問了這個程序,PHP會再啟動一個線程 ,flock只對B的線程起作用.A和B之間互不影響。A此時,A線程對文件的鎖定不影響 B對文件的操作!
有個
16200文本論壇,你從網路上搜一下,載下來參考一下它在數據操作上是怎麼做的。挺復雜的文件操作。
③ php中如何避免mysql資料庫死鎖
mysql一般不會死鎖,除非程序有問題。性能優先事務不優先的資料庫(設置)不要追求可靠性萬無一失。
網站性能問題主要是資料庫量大了以後,查詢掃描硬碟而產生的。其它性能不要太在意。編寫代碼的時候不要堅持性能原則,而是堅持可用性原則。初學者編寫代碼通常容易面向性能,但是一個項目的一個頁面幾百、幾千行代碼是很常見的。要面向可用性、可維護性、可讀性。這是項目原則。你看看java語言。對於網站,除了查詢掃描硬碟而產生的時間延遲,其它是不管的,只要不算有問題就可以。
連接方式是否為永久連接,在訪問量未達到高並發之前,還是非永久鏈接更好。非永久連接的資源消耗是不大於永久連接的,因為mysql是把連接許可權緩存的,不會多次掃描硬碟,性能是可執行級別的而不是查找數據級別的。在訪問量達到高並發之後,性能問題的原因是多方面的,多環節的,是否為永久連接不是主要原因。
④ thinkphp事務上鎖後整個資料庫都上鎖嗎
事務的鎖機制應該與thinkphp無關,主要看你採用什麼資料庫。
⑤ 並發時,php操作mysql需要加鎖嗎mysql在並發時讀寫數據表會加鎖,我覺得不用再手動加鎖,請問是這樣嗎
如果要一次性做很多次插入,最好在每個循環的前後加上LOCK TABLES和UNLOCK TABLES,從而讓其他進程也能訪問數據表;這么做性能依然不錯。
不然的話,會導致數據莫明其妙的插入不完整,或者沒插進去的情況。
⑥ php中mysql加鎖問題
有表鎖,行鎖,頁鎖
頁級:引擎 BDB。
表級:引擎 MyISAM , 理解為鎖住整個表,可以同時讀,寫不行
行級:引擎 INNODB , 單獨的一行記錄加鎖
1) 表級鎖:開銷小,加鎖快;不會出現死鎖;鎖定粒度大,發生鎖沖突的概率最高,並發度最低。
2) 行級鎖:開銷大,加鎖慢;會出現死鎖;鎖定粒度最小,發生鎖沖突的概率最低,並發度也最高。
3) 頁面鎖:開銷和加鎖時間界於表鎖和行鎖之間;會出現死鎖;鎖定粒度界於表鎖和行鎖之間,並發度一般。
一般不在PHP中使用鎖操作,因為如果鎖了庫,如果遇到錯誤沒有及時的解鎖,就會導致不能訪問數據的情況。
可以使用MYSQL的事務,就是定義事務開始,然後有幾個語句要執行,然後根據情況,如果有一個語句沒有執行成功,可以回滾(取消這幾個語句的執行),從而達到幾個語句都執行成功或者都不執行的效果,在強事務型的應用中一般使用這個方式
你可以去後盾人平台看看,裡面的東西不錯
⑦ PHP如何使用文件鎖解決高並發問題
<?php
//連接資料庫
$con=mysqli_connect("192.168.2.186","root","root","test");
//查詢商品數量是否大於0,大於0才能下單,並減少庫存
$fp = fopen("lock.txt", "r");
//加鎖
if(flock($fp,LOCK_EX))
{
$res=mysqli_fetch_assoc(mysqli_query($con,'SELECT total FROM shop WHERE id=1 LIMIT 1'));
if($res['total']>0){mysqli_query($con,'UPDATE shop SET total=total-1 WHERE id=1');}
//執行完成解鎖
flock($fp,LOCK_UN);
}
//關閉文件
fclose($fp);
unset($res);
mysqli_close($con);
?>
⑧ 開發php網站,mysql做資料庫一定要用鎖嗎
一般情況下,MYSQL在更新操作會自動加表鎖。不需要顯式加鎖。除非是數據要求嚴格的邏輯寫順序。
執行查詢語句(SELECT)前,會自動給涉及的所有表加讀鎖,在執行更新操作 (UPDATE、DELETE、INSERT等)前,會自動給涉及的表加寫鎖,這個過程並不需要用戶干預,因此,用戶一般不需要直接用LOCK TABLE命令給MyISAM表顯式加鎖。
⑨ PHP如何鎖住MYSQL表數據
引用
5
樓
xuzuning
的回復:只在
update
時才考慮加鎖比如添加一條數據,兩個用戶添加的數據ID相同當id為自增欄位時,這種情況不會發生。否則就不叫資料庫了+++++++
⑩ PHP MYSQL中 表鎖和行鎖 一般什麼情況下使用 另外具體怎麼寫
手動加鎖:
表級鎖
lock tables tablename read;//共享鎖
lock tables tablename write;//排它鎖
unlock tables;//解鎖
php直接query就行了。
除了MyIsam,誰還會無聊到顯式加鎖?
InnoDB直接跑事務默認會觸發隱式鎖,不需要自己lock和unlock。