session放入緩存(redis)、DB

爲何要把SESSION保存在緩存

就php來講,語言自己支持的session是以文件的方式保存到磁盤文件中,保存在指定的文件夾中,保存的路徑能夠在配置文件中設置或者在程序中使用函數session_save_path()進行設置,可是這麼作有弊端,php


第一就是保存到文件系統中,效率低,只要有用到session就會從好多個文件中查找指定的sessionid,效率很低。html

第二就是當用到多臺服務器的時候可能會出現,session丟失問題(實際上是保存在了其餘服務器上)。java

固然了,保存在緩存中能夠解決上面的問題,若是使用php自己的session函數,可使用session_set_save_handler()函數很方便的對session的處理過程進行從新控制。若是不用php的session系列函數,能夠本身編寫個相似的session函數,也是能夠的,我如今作的這個項目就是這樣,會根據用戶的mid、登陸時間進行求hash做爲sessionId,每次請求的時候都必須加上sessionId纔算合法(第一次登陸的時候是不須要的,這個時候會建立sessionId,返回給客戶端),這麼作也很方便、簡潔高效的。固然了,我這篇文章主要說的是在php自身的SESSION中」作作手腳」。mysql

SESSION保存在緩存中

php將緩存保存到redis中,可使用配置文件,對session的處理和保存作修改,固然了,在程序中使用ini_set()函數去修改也能夠,這個很方便測試,我這裏就使用這種方式,固然了,要是生產環境仍是建議使用配置文件。redis

<?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和瀏覽器請求的是同樣的,sql

JvaERnu.pngvyqyuq.png

是否是很方便呢,只須要改下配置文件就能夠實現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了,直接添加就好了

<?php
include_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( luluyrt@163.com )全部,未經做者本人贊成禁止任何形式的轉載,轉載文章以後必須在文章頁面明顯位置給出做者和原文鏈接,不然保留追究法律責任的權利。

相關文章
相關標籤/搜索