前言php
php默認使用文件存儲session,若是併發量大,效率會很是低。而redis對高併發的支持很是好,能夠利用redis替換文件來存儲session。html
最近就遇到了這個問題,以前找了網上的一套直播系統給客戶用,剛開始是沒問題的,在後麪人數上來以後網站開始變得卡頓,卡的一批。以後查看php慢日誌發現session_start()的身影,好吧,原來是萬惡的文件存儲session,跟我以前進的坑如出一轍……以前作的教務查詢系統直接用的session沒有用cookie,結果在高併發的狀況下php原地爆炸。mysql
[0x00007fff67ee6740] session_start() [0x00007fff67ee7b70] +++ dump failed
解決方案redis
坑中坑sql
由於這套直播系統一沒有用框架,二沒有設計規範,各類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。cookie
(如下代碼來源於網絡,也不知道原創是哪位大佬)網絡
編寫一個session管理類sessionManager.php,代碼以下:session
<?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的前兩行,成功接管,舒服。