前言
php默認使用文件存儲session,如果并發(fā)量大,效率會非常低。而redis對高并發(fā)的支持非常好,可以利用redis替換文件來存儲session。
最近就遇到了這個問題,之前找了網(wǎng)上的一套直播系統(tǒng)給客戶用,剛開始是沒問題的,在后面人數(shù)上來之后網(wǎng)站開始變得卡頓,卡的一批。之后查看php慢日志發(fā)現(xiàn)session_start()的身影,好吧,原來是萬惡的文件存儲session,跟我之前進的坑一模一樣……之前做的教務(wù)查詢系統(tǒng)直接用的session沒有用cookie,結(jié)果在高并發(fā)的情況下php原地爆炸。
[0x00007fff67ee6740] session_start() [0x00007fff67ee7b70] +++ dump failed
解決方案
坑中坑
因為這套直播系統(tǒng)一沒有用框架,二沒有設(shè)計規(guī)范,各種session操作散落在不同的文件里,用第一個解決方案完全屬于費力不討好。再者直播系統(tǒng)的聊天互動等功能已經(jīng)涉及大量的mysql操作,再用mysql接管session變相的增加了數(shù)據(jù)庫的壓力,最終確定了使用redis接管session。
具體實現(xiàn)
php有內(nèi)置的操作session的save_handler,使用session_set_save_handler,接管所有的session管理工作。在使用該函數(shù)前,先把php.ini配置文件的session.save_handler選項設(shè)置為user,否則session_set_save_handle不會生效。另外除了安裝redis之外,php擴展也需要增加redis。
(以下代碼來源于網(wǎng)絡(luò),也不知道原創(chuàng)是哪位大佬)
編寫一個session管理類sessionManager.php,代碼如下:
?php class SessionManager{ private $redis; private $sessionSavePath; private $sessionName; private $sessionExpireTime=30;//redis,session的過期時間為30s public function __construct(){ $this->redis = new Redis();//創(chuàng)建phpredis實例 $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為鍵,存儲 $this->redis->expire($id,$this->sessionExpireTime);//設(shè)置redis中數(shù)據(jù)的過期時間,即session的過期時間 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功能。因為本例中open、close和gc回調(diào)函數(shù)的作用不是很大,所以直接返回true。
在write回調(diào)函數(shù)中,以session ID 作為key,把session的數(shù)據(jù)作為value存儲到redis服務(wù)器,設(shè)置session的過期時間為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ù)。
使用時,只需包含SessionManager類,然后實例化一個SessionManager對象。
下面建立個session_set.php文件,代碼如下:
?php include('SessionManager.php'); new SessionManager(); $_SESSION['username'] = 'captain';
然后再創(chuàng)建一個session_get.php文件,代碼如下:
?php include('SessionManager.php'); new SessionManager(); echo $_SESSION['username'];
測試時,首先訪問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)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。