將PHP的session數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫中的代碼實(shí)例

字號(hào):


    這里我們將分享兩個(gè)將PHP的session數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫中的代碼實(shí)例,分別針對(duì)PostgreSQL與MySQL,需要的朋友可以參考下
    一個(gè)開發(fā)環(huán)境有多個(gè)網(wǎng)站,需要使用不同的session,解決方案很多。不過這次也高大上一把,用數(shù)據(jù)庫存,方便以后擴(kuò)展。
    PostgreSQL版
    首先是數(shù)據(jù)庫的部分
    --drop table php_session
    create unlogged table php_session
    (
      sess_id varchar(32) primary key,
      modify_time timestamp with time zone not null,
      sess_data varchar(3000) default ''
    );
    create index concurrently idx_php_session_modify_time on php_session(modify_time);
    --set_session(id, data)
    create or replace function set_session(varchar, varchar) returns void as $set_session$
      with upsert as (
        update php_session
        set modify_time = current_timestamp, sess_data = $2
        where sess_id = $1
        returning 1
      )
      insert into php_session (sess_id, modify_time, sess_data)
      select $1, current_timestamp, $2
      where not exists (
        select 1 from upsert
      );
    $set_session$ language sql;
    --get_session(id)
    create or replace function get_session(varchar) returns varchar as $get_session$
      select sess_data from php_session where sess_id = $1
    $get_session$ language sql;
    --del_session
    create or replace function del_session(varchar) returns void as $del_session$
      delete from php_session where sess_id = $1
    $del_session$ language sql;
    --gc_session
    create or replace function gc_session() returns void as $del_session$
      delete from php_session where modify_time < current_timestamp - interval '30 days'
    $del_session$ language sql;
    然后是PHP的部分
    <?php
    session_set_save_handler(
      function ($savePath, $sessionName) {//open
        return true;
      },
      function () {//close
        return true;
      },
      function ($id) {//read
        $sql = "select get_session($1)";
        $stmt = pg_query_params(SESSION_CONN, $sql, array($id));
        $result = pg_fetch_row($stmt);
         
        return $result[0];
      },
      function ($id, $data) {//write
        $sql = "select set_session($1, $2)";
        pg_query_params(SESSION_CONN, $sql, array($id, $data));
        return true;
      },
      function ($id) {//destroy
        $sql = "select del_session($1)";
        pg_query_params(SESSION_CONN, $sql, array($id, $data));
        return true;
      },
      function ($maxlifetime) {//gc
        //php needn't control the global session gc 
        return true;
      }
    );
    register_shutdown_function('session_write_close');
    ?>
    然后只要在session_start之前調(diào)用這個(gè)就可以了
    至于SESSION_CONN,那是我定義的一個(gè)常量,表示一個(gè)指向session數(shù)據(jù)庫的鏈接而已。
    MySQL版
    再總結(jié)一個(gè)針對(duì)MySQL的集成更多基礎(chǔ)功能的例子:
    表結(jié)構(gòu):
    CREATE TABLE IF NOT EXISTS `sessioninfo` (
    `sid` varchar(255) NOT NULL,
    `value` text NOT NULL,
    `expiration` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
    PRIMARY KEY (`sid`)
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
    session信息存儲(chǔ)到數(shù)據(jù)庫的類:
    class MySessionHandler implements SessionHandlerInterface {
      /**
      * @access private
      * @var object 數(shù)據(jù)庫連接
      */
      private $_dbLink;
      /**
      * @access private
      * @var string 保存session的表名
      */
      Private $_sessionTable;
      /**
      * @access private
      * @var string session名
      */
      private $_sessionName;
      /**
      * @const 過期時(shí)間
      */
      const SESSION_EXPIRE = 10;
      public function __construct($dbLink, $sessionTable) {
        if(!is_object($dbLink)) {
          return false;
        }
        $this->_dbLink = $dbLink;
        $this->_sessionTable = $sessionTable;
      }
      /**
      * 打開
      * @access public
      * @param string $session_save_path 保存session的路徑
      * @param string $session_name session名
      * @return integer
      */
      public function open($session_save_path, $session_name) {
        $this->_sessionName = $session_name;
        return 0;
      }
      /**
      * 關(guān)閉
      * @access public
      * @return integer
      */
      public function close() {
        return 0;
      }
      /**
      * 關(guān)閉session
      * @access public
      * @param string $session_id session ID
      * @return string
      */
      public function read($session_id) {
        $query = "SELECT value FROM {$this->_sessionTable} WHERE sid = {$session_id} AND UNIX_TIMESTAMP(expiration) + " . self::SESSION_EXPIRE . " > UNIX_TIMESTAMP(NOW())";
        $result = $this->_dbLink->query($query);
        if(!isset($value) || empty($value)) {
          $value = "";
          return $value;
        }
        $this->_dbLink->query("UPDATE {$this->_sessionTable} SET expiration = CURRENT_TIMESTAMP() WHERE sid = {$session_id}");
        $value = $result->fetch_array();
        $result->free();
        return $value['value'];
      }
      /**
      * 寫入session
      * @access public
      * @param string $session_id session ID
      * @param string $session_data session data
      * @return integer
      */
      public function write($session_id, $session_data) {
        $query = "SELECT value FROM {$this->_sessionTable} WHERE sid = '{$session_id}' AND UNIX_TIMESTAMP(expiration) + " . self::SESSION_EXPIRE . " > UNIX_TIMESTAMP(NOW())";
        $result = $this->_dbLink->query($query);
        $result = $result->fetch_array();
        if(!empty($result)) {
          $result = $this->_dbLink->query("UPDATE {$this->_sessionTable} SET value = {$session_data} WHERE sid = {$session_id}");
        }
        else{
          $result = $this->_dbLink->query("INSERT INTO {$this->_sessionTable} (sid, value) VALUES ('{$session_id}', '{$session_data}')");
        }
        if($result){
          return 0;
        }
        else{
          return 1;
        }    
      }
      /**
      * 銷魂session
      * @access public
      * @param string $session_id session ID
      * @return integer
      */
      public function destroy($session_id) {
        $result = $this->_dbLink->query("DELETE FROM {$this->_sessionTable} WHERE sid = '{$session_id}'");
        if($result){
          return 0;
        }
        else{
          return 1;
        }
      }
      /**
      * 垃圾回收
      * @access public
      * @param string $maxlifetime session 最長生存時(shí)間
      * @return integer
      */
      public function gc($maxlifetime) {
        $result = $this->_dbLink->query("DELETE FROM {$this->_sessionTable} WHERE UNIX_TIMESTAMP(expiration) < UNIX_TIMESTAMP(NOW()) - " . self::SESSION_EXPIRE);
        if($result){
          return 0;
        }
        else{
          return 1;
        }
      }
    }
    -------------------------------------------------------------------------------
    $dbLink = new mysqli("localhost", "root", "root", "test");
    $sessionTable = "sessioninfo";
    $handler = new MySessionHandler($dbLink, $sessionTable);
    session_set_save_handler($handler);
    session_start();
    $_SESSION['name'] = "test";
    echo $_SESSION["name"];
    //session_destroy();