phprediskey
本文介紹了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實現redis資料庫指定庫號遷移的方法
這篇文章主要介紹了php實現redis資料庫指定庫號遷移的方法,涉及對於redis資料庫的操作技巧,非常具有實用價值,需要的朋友可以參考下
本文實例講述了php實現redis資料庫指定庫號遷移的方法,分享給大家供大家參考。具體如下:
redis普通的資料庫遷移,只能整個redis
save,或者利用主從,當然也可以安裝一個redis-mp,不過比較麻煩,這里提供一種php的腳本,實現指定庫號的遷移,其實也就是遍歷根據存儲類型,讀出來,插入新庫,效果是這樣:
代碼如下:
[root@localhost
~]#
php
1.php
1/407
101/407
201/407
301/407
401/407
PHP實例代碼如下:
代碼如下:
<?php
$from
=
'10.0.2.52:6379/7';
$to
=
'127.0.0.1:6379/7';
$from_redis
=
redis_init($from);
$to_redis
=
redis_init($to);
$keys
=
$from_redis->keys('*');
$count
=
0;
$total
=
count($keys);
foreach($keys
as
$key){
if(++$count
%
100
==
1){
echo
"$count/$totaln";
}
$type
=
$from_redis->type($key);
switch($type){
case
Redis::REDIS_STRING:
$val
=
$from_redis->get($key);
$to_redis->set($key,
$val);
break;
case
Redis::REDIS_LIST:
$list
=
$from_redis->lRange($key,
0,
-1);
foreach($list
as
$val){
$to_redis->rPush($key,
$val);
}
break;
case
Redis::REDIS_HASH:
$hash
=
$from_redis->hGetAll($key);
$to_redis->hMSet($key,
$hash);
break;
case
Redis::REDIS_ZSET:
$zset
=
$from_redis->zRange($key,
0,
-1,
true);
foreach($zset
as
$val=>$score){
$to_redis->zAdd($key,
$score,
$val);
}
break;
}
}
function
redis_init($conf){
$redis
=
new
Redis();
preg_match('/^([^:]+)(:[0-9]+)?/(.+)?/',
$conf,
$ms);
$host
=
$ms[1];
$port
=
trim($ms[2],
':');
$db
=
$ms[3];
$redis->connect($host,
$port);
$redis->select($db);
return
$redis;
}
?>
希望本文所述對大家的php程序設計有所幫助。
⑶ php redis Hash 怎麼通過 一個指定的value 查找到對應的 key 值
phpredis是php的一個擴展,效率是相當高有鏈表排序功能,對創建內存級的模塊業務關系很有用;
如果對系統存儲使用的數據以兩種角度分類,一種是按數據的大小劃分,分成大數據和小數據,另一種是按數據的冷熱程度劃分,分成冷數據和熱數據,熱數據是指讀或寫比較頻繁的數據,反之則是冷數據。
可以舉一些具體的例子來說明數據的大小和冷熱屬性。比如網站總的注冊用戶數,這明顯是一個小而熱的數據,小是因為這個數據只有一個值,熱是因為注冊用戶數隨時間變化很頻繁。再比如,用戶最新訪問時間數據,這是一個量比較大,冷熱不均的數據,大是數據的粒度是用戶級別,每一個用戶都有數據,如果有一千萬用戶,就意味著有一千萬的數據,冷熱不均是因為活躍用戶的最新訪問時間變化很頻繁,但是可能有很大一部非活躍用戶訪問時間長時間不會發生變化。
大體而言,Redis 最適合處理的是小而熱,而且是寫頻繁,或者讀寫都比較頻繁的熱數據。對於大而熱的數據,如果其它方式很難解決問題,也可以考慮使用 Redis 解決,但是一定要非常謹慎,防止數據無限膨脹。原因如下:
首先,對於冷數據,無論大小,都不建議放在 Redis 中。Redis 數據要全部放在內存中,資源寶貴,把冷數據放在其中實在是一種浪費,冷數據放在普通的存儲比如關系資料庫中就好了。
其次,對於熱數據,尤其是寫頻繁的熱數據,如果量比較小,是最適合放到 Redis 中的。比如上面提到的網站總的注冊用戶數,就是典型的 Redis 用做計數器的例子。再比如論壇最新發表列表,最新報名列表,可以控制數量在幾百到一千的規模,也是典型的 redis 做最新列表的使用方式。
另外,對於量比較大的熱數據(或者冷熱不均數據),使用 Redis 時一定要比較謹慎。這種類型數據很容易引起數據膨脹,導致 Redis 消耗內存巨大,讓系統難以承受。薄荷的一個慘痛教訓是把用戶關注(以及被關注)數據放在 Redis 中,這是一種數據量極大,冷熱很不均衡的數據,在幾百萬的用戶級別就佔用了近 10 GB左右內存,讓 Redis 變得難以應付。應對這種類型的數據,可以用普通存儲 + 緩存的方式。
如果用對了地方,比如在小而熱的數據情形,Redis 表現很棒,如果用錯了地方,Redis 也會帶來昂貴的代價,所以使用時務必謹慎。
⑷ php怎麼查看redis的key
首先開啟redis服務:
打開cmd命令窗口,將根目錄切換至redis根目錄;
開啟redis服務:輸入命令:redis-server.exe redis.conf 然後回車。
開啟成功後,此窗口不要關閉,另打開一個cmd窗口,連接客戶端:
1.打開cmd命令窗口,將根目錄切換至redis根目錄;
2.輸入連接命令:redis-cli.exe -h 127.0.0.1 -p 6379 然後回車
3.輸入key的查看命令:keys *
⑸ 為什麼php調用redis 返回+ok
Redis::__construct構造函數
$redis = new Redis();
connect, open 鏈接redis服務
參數
host: string,服務地址
port: int,埠號
timeout: float,鏈接時長 (可選, 默認為 0 ,不限鏈接時間)
注: 在redis.conf中也有時間,默認為300
pconnect, popen 不會主動關閉的鏈接
參考上面
setOption 設置redis模式
getOption 查看redis設置的模式
ping 查看連接狀態
KEY相關操作
DEL
移除給定的一個或多個key。
如果key不存在,則忽略該命令。
時間復雜度:
O(N),N為要移除的key的數量。
移除單個字元串類型的key,時間復雜度為O(1)。
移除單個列表、集合、有序集合或哈希表類型的key,時間復雜度為O(M),M為以上數據結構內的元素數量。
返回值:
被移除key的數量。這樣呢我自己學習在後盾人看見的,老師講的很詳細,希望對你有用😊(ง •̀_•́)ง努力
⑹ php redis有沒有辦法獲取所有redis列表
首先需要定時清理的是不是可以設置下有效時間呢,接著你說的獲取redis全部key*是想在php中操作么,那麼建議
$redis->keys('*');
這樣解決就可以了,簡單粗暴,個人建議還是去後盾網去經常看看教學視頻學習學習吧
⑺ PHP 讀取redis 時,key 的長短會影響讀取時間嗎
redis key 的長度主要會影響空間佔用,時間上,差距可以忽略
所以這個不用擔心了
⑻ 如何測試 php安裝成功redis
解壓安裝:
復制代碼代碼如下:
tar -xvf redis-2.8.17.tar.gz
make
sudo make install
為方便使用,在/usr目錄下創建redis目錄,講如下幾個文件拷貝到/usr/redis/目錄下:
復制代碼代碼如下:
/yourdir/redis-2.8.17/redis.conf
/yourdir/redis-2.8.17/src/redis-benchmark
/yourdir/redis-2.8.17/src/redis-server
/yourdir/redis-2.8.17/src/redis-cli
當然,你也可以通過軟連接的方式達到方便使用的目的。此外,你也可以將redis-server加入開機啟動,此處從略。
2.redis測試
1)先開啟redis服務端程序
為方便測試,我們將redis.conf配置文件中的loglevel和logfile的值,修改後如下:
loglevel debug
logfile 「/tmp/redis.log」
jay13@ubuntu:/usr/redis$ redis-server redis.conf
2)開啟redi客戶端,通過客戶端向redis資料庫中進行增刪改查操作。整個操作過程中生成的日誌可以到/tmp/redis.log中查看。
以最簡單的key操作為例,實例如下:
復制代碼代碼如下:
jay13@ubuntu:/usr/redis$ redis-cli
127.0.0.1:6379> set jay13 jb51.net
OK
127.0.0.1:6379> set jay hello,world
OK
127.0.0.1:6379> get jay
"hello,world"
127.0.0.1:6379> get jay13
"jb51.net"
127.0.0.1:6379> del jay
(integer) 1
127.0.0.1:6379> get jay
(nil)
127.0.0.1:6379> set jay13 www.jb51.net
OK
127.0.0.1:6379> get jay13
"www.jb51.net"
3.安裝phpredis擴展
在使用sudo apt-get install php5安裝php時,默認是沒有安裝phpize的,我們安裝phpredis時,需要用到phpize,因此,需要先安裝phpize。
1)我們通過安裝php開發者工具來獲取phpize。執行如下命令即可:
復制代碼代碼如下:
sudo apt-get install php5-dev
2)獲取phpredis源文件
按照GitHub上的說明進行如下安裝時,
復制代碼代碼如下:
phpize
./configure --enable-redis-igbinary
make && make install
可能會出現如下出錯說明:
復制代碼代碼如下:
checking for igbinary includes... configure: error: Cannot find igbinary.h