標籤:
為什麼要把SESSION儲存在緩衝
就php來說,語言本身支援的session是以檔案的方式儲存到磁碟檔案中,儲存在指定的檔案夾中,儲存的路徑可以在設定檔中設定或者在程式中使用函數session_save_path()進行設定,但是這麼做有弊端,
第一就是儲存到檔案系統中,效率低,只要有用到session就會從好多個檔案中尋找指定的sessionid,效率很低。
第二就是當用到多台伺服器的時候可能會出現,session丟失問題(其實是儲存在了其他伺服器上)。
當然了,儲存在緩衝中可以解決上面的問題,如果使用php本身的session函數,可以使用session_set_save_handler()函數很方便的對session的處理過程進行重新控制。如果不用php的session系列函數,可以自己編寫個類似的session函數,也是可以的,我現在做的這個項目就是這樣,會根據使用者的mid、登入時間進行求hash作為sessionId,每次請求的時候都必須加上sessionId才算合法(第一次登入的時候是不需要的,這個時候會建立sessionId,返回給用戶端),這麼做也很方便、簡潔高效的。當然了,我這篇文章主要說的是在php自身的SESSION中”做做手腳”。
SESSION儲存在緩衝中
php將緩衝儲存到redis中,可以使用設定檔,對session的處理和儲存做修改,當然了,在程式中使用ini_set()函數去修改也可以,這個很方便測試,我這裡就使用這種方式,當然了,要是生產環境還是建議使用設定檔。
<?phpini_set("session.save_handler", "redis");ini_set("session.save_path", "tcp://localhost:6379");session_start();header("Content-type:text/html;charset=utf-8");if(isset($_SESSION[‘view‘])){ $_SESSION[‘view‘] = $_SESSION[‘view‘] + 1;}else{ $_SESSION[‘view‘] = 1;}echo "【view】{$_SESSION[‘view‘]}";
這裡設定session.save_handler方式為redis,session.save_path為redis的地址和連接埠,設定之後重新整理,再回頭查看redis,會發現redis中的產生了sessionId,sessionId和瀏覽器請求的是一樣的,
是不是很方便呢,只需要改下設定檔就可以實現redis中儲存session,但是我這裡要說的是通過程式的方式來處理session儲存到redis或者db,下面一起來看看。
通過php提供的介面,自己改寫session的處理函數
這裡可以先看看php的這個函數session_set_save_handler,php5.4及之後可以直接實現SessionHandlerInterface介面,代碼會更加簡潔。重寫的時候主要有下面幾個方法
open(string $savePath, string $sessionName); //open類似於建構函式,開始會話的時候會調用,比如使用session_start()函數之後
close(); //類似於類的解構函式,在write函數調用之後調用,session_write_close()之後之後也會執行
read(string $sessionId); //讀取session的時候調用
write(string $sessionId, string $data); //儲存資料的時候調用
destory($sessionId); //銷毀會話的時候(session_destory()或者session_regenerate_id())會調用
gc($lifeTime); //垃圾清理函數,清理掉到期作廢的資料
主要就是實現這幾個方法,根據不同的儲存驅動可以自己設定不同的具體方法,我實現了mysql資料庫和redis這兩種儲存session的驅動,如果有需要的話可以自己去擴充,擴充很方便很容易。
下面是我的redis的實現(db和redis差不多,redis代碼少,貼出來):
我使用了介面的方式,這樣擴充起來更方便,那天想用memcached了,直接添加就行了
<?phpinclude_once __DIR__."/interfaceSession.php";/** * 以db的方式儲存session */class redisSession implements interfaceSession{ /** * 儲存session的資料庫表的資訊 */ private $_options = array( ‘handler‘ => null, //資料庫連接控制代碼 ‘host‘ => null, ‘port‘ => null, ‘lifeTime‘ => null, ); /** * 建構函式 * @param $options 設定資訊數組 */ public function __construct($options=array()){ if(!class_exists("redis", false)){ die("必須安裝redis擴充"); } if(!isset($options[‘lifeTime‘]) || $options[‘lifeTime‘] <= 0){ $options[‘lifeTime‘] = ini_get(‘session.gc_maxlifetime‘); } $this->_options = array_merge($this->_options, $options); } /** * 開始使用該驅動的session */ public function begin(){ if($this->_options[‘host‘] === null || $this->_options[‘port‘] === null || $this->_options[‘lifeTime‘] === null ){ return false; } //設定session處理函數 session_set_save_handler( array($this, ‘open‘), array($this, ‘close‘), array($this, ‘read‘), array($this, ‘write‘), array($this, ‘destory‘), array($this, ‘gc‘) ); } /** * 自動開始回話或者session_start()開始回話後第一個調用的函數 * 類似於建構函式的作用 * @param $savePath 預設的儲存路徑 * @param $sessionName 預設的參數名,PHPSESSID */ public function open($savePath, $sessionName){ if(is_resource($this->_options[‘handler‘])) return true; //串連redis $redisHandle = new Redis(); $redisHandle->connect($this->_options[‘host‘], $this->_options[‘port‘]); if(!$redisHandle){ return false; } $this->_options[‘handler‘] = $redisHandle; $this->gc(null); return true; } /** * 類似於解構函式,在write之後調用或者session_write_close()函數之後調用 */ public function close(){ return $this->_options[‘handler‘]->close(); } /** * 讀取session資訊 * @param $sessionId 通過該Id唯一確定對應的session資料 * @return session資訊/空串 */ public function read($sessionId){ return $this->_options[‘handler‘]->get($sessionId); } /** * 寫入或者修改session資料 * @param $sessionId 要寫入資料的session對應的id * @param $sessionData 要寫入的資料,已經序列化過了 */ public function write($sessionId, $sessionData){ return $this->_options[‘handler‘]->setex($sessionId, $this->_options[‘lifeTime‘], $sessionData); } /** * 主動銷毀session會話 * @param $sessionId 要銷毀的會話的唯一id */ public function destory($sessionId){ return $this->_options[‘handler‘]->delete($sessionId) >= 1 ? true : false; } /** * 清理繪畫中的到期資料 * @param 有效期間 */ public function gc($lifeTime){ //擷取所有sessionid,讓到期的釋放掉 $this->_options[‘handler‘]->keys("*"); return true; }}
看看簡單原廠模式
class session { /** * 驅動程式控制代碼儲存 */ private static $_handler = null; /** * 建立session驅動程式 */ public static function getSession($type, $options){ //單例 if(isset($handler)){ return self::$_handler; } switch ($type) { case ‘db‘: //資料庫驅動session類型 include_once __DIR__."/driver/dbSession.php"; $handler = new dbSession($options); break; case ‘redis‘: //redis驅動session類型 include_once __DIR__."/driver/redisSession.php"; $handler = new redisSession($options); break; default: return false; break; } return self::$_handler = $handler; }}
調用也很簡單,
session::getSession(‘redis‘,array( ‘host‘ => "localhost", ‘port‘ => "6379", ))->begin();session_start();
資料庫版本的也一樣很簡單就可以配置,需要的話可以在這裡下載完整版和demo
本文著作權歸作者iforever([email protected])所有,未經作者本人同意禁止任何形式的轉載,轉載文章之後必須在文章頁面明顯位置給出作者和原文串連,否則保留追究法律責任的權利。
session儲存到緩衝(redis)、DB