PHP中使用Redis接管檔案儲存Session詳解
前言
php預設使用檔案儲存session,如果併發量大,效率會非常低。而redis對高併發的支援非常好,可以利用redis替換檔案來儲存session。
最近就遇到了這個問題,之前找了網上的一套直播系統給客戶用,剛開始是沒問題的,在後麵人數上來之後網站開始變得卡頓,卡的一批。之後檢視php慢日誌發現session_start()的身影,好吧,原來是萬惡的檔案儲存session,跟我之前進的坑一模一樣……之前做的教務查詢系統直接用的session沒有用cookie,結果在高併發的情況下php原地爆炸。
[0x00007fff67ee6740] session_start() [0x00007fff67ee7b70] +++ dump failed
解決方案
- 將session全面更換為cookie
- 使用mysql或redis接管session
坑中坑
因為這套直播系統一沒有用框架,二沒有設計規範,各種session操作散落在不同的檔案裡,用第一個解決方案完全屬於費力不討好。再者直播系統的聊天互動等功能已經涉及大量的mysql操作,再用mysql接管session變相的增加了資料庫的壓力,最終確定了使用redis接管session。
具體實現
php有內建的操作session的save_handler,使用session_set_save_handler,接管所有的session管理工作。在使用該函式前,先把php.ini配置檔案的session.save_handler選項設定為user,否則session_set_save_handle不會生效。另外除了安裝redis之外,php擴充套件也需要增加redis。
(以下程式碼來源於網路,也不知道原創是哪位大佬)
編寫一個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();//建立phpredis例項 $this->redis->connect('127.0.0.1',6379);//連線redis $this->redis->auth("107lab");//授權 $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);//設定redis中資料的過期時間,即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建構函式主要用來連線Redis伺服器,使用session_set_save_handler函式設定session回撥函式,並呼叫session_start函式開啟session功能。因為本例中open、close和gc回撥函式的作用不是很大,所以直接返回true。
在write回撥函式中,以session ID 作為key,把session的資料作為value儲存到redis伺服器,設定session的過期時間為30秒。在read回撥函中,以session ID 作為key從redis伺服器中讀取資料,並返回此資料。而在destroy回撥函式重,則以session ID 作為key 從redis伺服器中刪除對應的session資料。
使用時,只需包含SessionManager類,然後例項化一個SessionManager物件。
下面建立個session_set.php檔案,程式碼如下:
<?php include('SessionManager.php'); new SessionManager(); $_SESSION['username'] = 'captain';
然後再建立一個session_get.php檔案,程式碼如下:
<?php include('SessionManager.php'); new SessionManager(); echo $_SESSION['username'];
測試時,首先訪問session_set.php,然後再訪問session_get.php,輸出結果如下所示:
再檢視redis資料庫,如下所示:
127.0.0.1:6379> keys * 1) "oe94eic337slnjv1bvlreoa574" 127.0.0.1:6379> get oe94eic337slnjv1bvlreoa574 "username|s:7:\"captain\";"
測試完美~
然後將原系統中的session_start()替換成session_set.php的前兩行,成功接管,舒服。