前言
php默認(rèn)使用文件存儲(chǔ)session,如果并發(fā)量大,效率會(huì)非常低。而redis對高并發(fā)的支持非常好,可以利用redis替換文件來存儲(chǔ)session。
最近就遇到了這個(gè)問題,之前找了網(wǎng)上的一套直播系統(tǒng)給客戶用,剛開始是沒問題的,在后面人數(shù)上來之后網(wǎng)站開始變得卡頓,卡的一批。之后查看php慢日志發(fā)現(xiàn)session_start()的身影,好吧,原來是萬惡的文件存儲(chǔ)session,跟我之前進(jìn)的坑一模一樣……之前做的教務(wù)查詢系統(tǒng)直接用的session沒有用cookie,結(jié)果在高并發(fā)的情況下php原地爆炸。
[0x00007fff67ee6740] session_start()
[0x00007fff67ee7b70] +++ dump failed
解決方案
- 將session全面更換為cookie
- 使用mysql或redis接管session
坑中坑
因?yàn)檫@套直播系統(tǒng)一沒有用框架,二沒有設(shè)計(jì)規(guī)范,各種session操作散落在不同的文件里,用第一個(gè)解決方案完全屬于費(fèi)力不討好。再者直播系統(tǒng)的聊天互動(dòng)等功能已經(jīng)涉及大量的mysql操作,再用mysql接管session變相的增加了數(shù)據(jù)庫的壓力,最終確定了使用redis接管session。
具體實(shí)現(xiàn)
php有內(nèi)置的操作session的save_handler,使用session_set_save_handler,接管所有的session管理工作。在使用該函數(shù)前,先把php.ini配置文件的session.save_handler選項(xiàng)設(shè)置為user,否則session_set_save_handle不會(huì)生效。另外除了安裝redis之外,php擴(kuò)展也需要增加redis。
(以下代碼來源于網(wǎng)絡(luò),也不知道原創(chuàng)是哪位大佬)
編寫一個(gè)session管理類sessionManager.php,代碼如下:
?php
class SessionManager{
private $redis;
private $sessionSavePath;
private $sessionName;
private $sessionExpireTime=30;//redis,session的過期時(shí)間為30s
public function __construct(){
$this->redis = new Redis();//創(chuàng)建phpredis實(shí)例
$this->redis->connect('127.0.0.1',6379);//連接redis
$this->redis->auth("107lab");//授權(quán)
$retval = session_set_save_handler(
array($this,"open"),
array($this,"close"),
array($this,"read"),
array($this,"write"),
array($this,"destroy"),
array($this,"gc")
);
session_start();
}
public function open($path,$name){
return true;
}
public function close(){
return true;
}
public function read($id){
$value = $this->redis->get($id);//獲取redis中的指定記錄
if($value){
return $value;
}else{
return '';
}
}
public function write($id,$data){
if($this->redis->set($id,$data)){//以session ID為鍵,存儲(chǔ)
$this->redis->expire($id,$this->sessionExpireTime);//設(shè)置redis中數(shù)據(jù)的過期時(shí)間,即session的過期時(shí)間
return true;
}
return false;
}
public function destroy($id){
if($this->redis->delete($id)){//刪除redis中的指定記錄
return true;
}
return false;
}
public function gc($maxlifetime){
return true;
}
public function __destruct(){
session_write_close();
}
}
SessionManager構(gòu)造函數(shù)主要用來連接Redis服務(wù)器,使用session_set_save_handler函數(shù)設(shè)置session回調(diào)函數(shù),并調(diào)用session_start函數(shù)開啟session功能。因?yàn)楸纠衞pen、close和gc回調(diào)函數(shù)的作用不是很大,所以直接返回true。
在write回調(diào)函數(shù)中,以session ID 作為key,把session的數(shù)據(jù)作為value存儲(chǔ)到redis服務(wù)器,設(shè)置session的過期時(shí)間為30秒。在read回調(diào)函中,以session ID 作為key從redis服務(wù)器中讀取數(shù)據(jù),并返回此數(shù)據(jù)。而在destroy回調(diào)函數(shù)重,則以session ID 作為key 從redis服務(wù)器中刪除對應(yīng)的session數(shù)據(jù)。
使用時(shí),只需包含SessionManager類,然后實(shí)例化一個(gè)SessionManager對象。
下面建立個(gè)session_set.php文件,代碼如下:
?php
include('SessionManager.php');
new SessionManager();
$_SESSION['username'] = 'captain';
然后再創(chuàng)建一個(gè)session_get.php文件,代碼如下:
?php
include('SessionManager.php');
new SessionManager();
echo $_SESSION['username'];
測試時(shí),首先訪問session_set.php,然后再訪問session_get.php,輸出結(jié)果如下所示:
再查看redis數(shù)據(jù)庫,如下所示:
127.0.0.1:6379> keys *
1) "oe94eic337slnjv1bvlreoa574"
127.0.0.1:6379> get oe94eic337slnjv1bvlreoa574
"username|s:7:\"captain\";"
測試完美~
然后將原系統(tǒng)中的session_start()替換成session_set.php的前兩行,成功接管,舒服。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:- PHP使用Redis實(shí)現(xiàn)Session共享的實(shí)現(xiàn)示例
- PHP SESSION機(jī)制的理解與實(shí)例
- 實(shí)現(xiàn)PHP中session存儲(chǔ)及刪除變量
- PHP實(shí)現(xiàn)用session來實(shí)現(xiàn)記錄用戶登陸信息
- PHP實(shí)現(xiàn)負(fù)載均衡session共享redis緩存操作示例
- php session_decode函數(shù)用法講解