redis/分佈式文件存儲系統/數據庫 存儲session,解決負載均衡集羣中session不一致問題

先來講下session和cookie的異同php

 

session和cookie不只僅是一個存放在服務器端,一個存放在客戶端那麼籠統mysql

session雖然存放在服務器端,可是也須要和客戶端相互匹配,試想一個瀏覽器爲啥session老是同樣的(過時或者關閉不算),主要得益於在瀏覽器端有個cook,名字叫"PHPSESSID"這個cookie裏面就是一串字符串。這個字符串就是用於標示session的,在使用session時當服務器端發現這個cookie後就會到服務器端session文件存放目錄查找名稱爲"sess_PHPSESSID值" 的文件(沒有就建立之), 這個文件裏面就是存放的session的一些數據(序列化後的數據)面試

因此,即便你把這個文件刪掉了,下次再使用session它又會從新建立一個一樣名稱的文件,固然要是把那個cookie給刪掉了,那就得從新命名了redis

session 默認狀況下是存放在每臺服務器本地目錄的,在'php.ini'有相應配置sql

 

服務器端配置:數據庫

session.save_handler = files    (默認爲file,定義session在服務端的保存方式,file意爲把sesion保存到一個臨時文件裏,若是咱們想自定義別的方式保存,好比數據庫之類需設置爲'user')數組

 

session.save_path = "D:/wamp/php/sessiondata/"   (定義服務端存儲session的臨時文件的位置)瀏覽器

 

session.auto_start = 0  (如置1,則不用在每一個文件裏寫session_start(); session自動start :)bash

 

session.gc_probability = 1

session.gc_divisor    = 100

session.gc_maxlifetime = 1440(以上三個構成session的垃圾自動回收機制,session.gc_probability與session.gc_divisor構成執行session清理的機率,理論上的解釋爲服務端按期有必定的機率調用gc函數來對session進行清理,清理的機率爲:gc_probability/gc_divisor 好比:1/100  表示每個新會話初始化時,有1%的機率會被垃圾回收機制回收,清理的標準爲 session.gc_maxlifetime 定義的時間)服務器

 

還有些客戶端相關的配置

session.use_cookies = 1  (sessionid在客戶端採用的存儲方式,置1表明使用cookie記錄客戶端的sessionid,同時,$_COOKIE變量裏纔會有$_COOKIE['PHPSESSIONID']這個cookie存在

 

session.use_only_cookies = 1  (也是定義sessionid在客戶端採用的存儲方式,置1表明僅僅使用 cookie 來存放會話 ID)

 

session.use_trans_sid = 0   (對應於上面那個設置,這裏若是置1,則表明容許sessionid經過url參數傳遞,同理,建議設置成0, 因此這裏糾正下一些面試題什麼的 禁用cookie是否可以使用session, 答案是固然可以只要把該值設置爲1)

 

session.referer_check =   (這個設置在session.use_trans_sid = 1的時候纔會生效,目的是檢查HTTP頭中的"Referer"以判斷包含於URL中的會話id是否有效,HTTP_REFERER必須包含這個參數指定的字符串,不然URL中的會話id將被視爲無效。因此通常默認爲空,即不檢查)

 

session.name = PHPSESSID   (定義sessionid的名稱,即變量名,因此經過瀏覽器http工具看到的http頭文件裏的PHPSESSID=##############)

 

session.cookie_lifetime = 0   (保存sessionid的cookie文件的生命週期,如置0,表明會話結束,則sessionid就自動消失,常見的強行關閉瀏覽器,就會丟失上一次的sessionid)

 

因此,經過上面咱們能夠知道,默認狀況下session是存放在每臺服務器本地的,所以在集羣環境下若是要使用session ,若是使用默認配置的話會出問題的,就是剛剛客戶訪問A服務器session文件存在A上面,但過一會可能會分配給該客戶B服務器,這時B服務器上這個文件不存在,數據也就丟失了。

 

即如此,那麼解決問題的辦法就是把session存放到單獨的服務器上,要麼數據庫,要麼redis, 要麼文件服務器

筆者這裏一一說明設置方法

 

1、使用redis存放session

這個筆者只說最簡單的,不採用不少人用的還要寫個PHP類規定怎樣存放(固然也能夠這麼作,若是在某些特殊需求狀況下)

先修改php.ini 配置

session.save_handler = redis
session.save_path = "tcp://127.0.0.1:6379"

固然了,也能夠在php程序中設置

ini_set('session.save_handler','redis');
ini_set('session.save_path','tcp://127.0.0.1:6379');

若是你的redis裏面配置了密碼,能夠這樣設置

session.save_handler = redis
session.save_path = "tcp://127.0.0.1:6379?auth=authpwd"

 

2、使用文件服務器存放session

這個筆者以爲比較簡單,筆者公司裏面直接把分佈式文件服務器掛載到指定目錄下,而後訪問分佈式文件服務器就像訪問本地文件夾同樣,這裏只須要設置下 保存路徑便可

session.save_path = "xxxx"

 

3、使用數據庫存放session

這個略顯複雜,要寫個PHP類,指定如何打開、讀取、寫入、銷燬、GC垃圾回收、關閉,不過筆者不懶仍是手動寫一個意思意思

<?php 
class sessionHandler{
    /**
    * session 存放的庫
    */
    const SESSION_DB = 'mytest';

    /**
    * session 存放的表
    */
    const SESSION_TABLE = 'session';

    /**
    * @var string $_dbHandler 數據庫連接句柄
    */
    private $_dbHandler;

    /**
    * @var string $_dbHost 數據庫主機
    */
    private $_dbHost;

    /**
    * @var string $_dbUser 數據庫用戶名
    */
    private $_dbUser;

    /**
    * @var string $_dbUser 數據庫密碼
    */
    private $_dbPasswd;

    /**
    * @var string $_name session 名稱
    */
    private $_name;

    /**
    * 構造函數
    * @param string $dbHost 數據庫主機
    * @param string $dbUser 數據庫用戶名
    * @param string $dbPasswd 數據庫密碼
    * @return void
    */
    public function __construct($dbHost, $dbUser, $dbPasswd)
    {
        $this->_dbHost = $dbHost;
        $this->_dbUser = $dbUser;
        $this->_dbPasswd = $dbPasswd;
    }

    /**
    * 連接數據庫
    * @param string $savePath 存儲路徑
    * @param string $name 名稱
    * @return boolean
    */
    public function open($savePath, $name)
    {
        $this->_dbHandler = mysql_connect($this->_dbHost, $this->_dbUser, $this->_dbPasswd);
        if(!$this->_dbHandler)
        {
            return false;
        }
        $this->_name = $name;
        mysql_select_db(self::SESSION_DB, $this->_dbHandler);
        return true;
    }

    /**
    * 讀session
    * @param string $sessionId session id
    * @return mixd 存在返回數組  不然返回空
    */
    public function read($sessionId)
    {
        $data = '';
        $sql = sprintf('SELECT `data` FROM ' . self::SESSION_TABLE . ' WHERE `id`="%s"', $sessionId);
        $result = mysql_query($sql, $this->_dbHandler);
        if(mysql_num_rows($result) == 1)
        {
            list($data) = mysql_fetch_array($result, MYSQL_NUM);
        }
        return $data;
    }

    /**
    * 連接數據庫
    * @param string $sessionId session id
    * @param string $data 數據
    * @return boolean
    */
    public function write($sessionId, $data)
    {
        $sql = sprintf(
                'REPLACE INTO 
                ' . self::SESSION_TABLE . ' (`id`, `data`, `last_time`) 
                VALUES 
                ("%s", "%s", %d)', 
                $sessionId,     
                mysql_escape_string($data),  
                time()
                );
        mysql_query($sql, $this->_dbHandler);
        return mysql_affected_rows($this->_dbHandler) > 0;
    }

    /**
    * 連接數據庫
    * @param int $expire 生存週期
    * @return boolean
    */
    public function gc($expire)
    {
        $sql = sprintf(
                        'DELETE FROM `' . self::SESSION_TABLE . '`
                        WHERE 
                        `last_time` < NOW() - %d',
                        $expire
                    );
        mysql_query($sql, $this->_dbHandler);
        return mysql_affected_rows($this->_dbHandler) > 0;
    }

    /**
    * 關閉數據庫連接
    * @param void
    * @return boolean
    */
    public function close()
    {
        return mysql_close($this->_dbHandler);
    }

    /**
    * 銷燬session
    * @param string $sessionId
    * @return boolean
    */
    public function destroy($sessionId)
    {
        $sql = sprintf('DELETE FROM `' . self::SESSION_TABLE . '` WHERE `id`="%s"', $sessionId);
        mysql_query($sql, $this->_dbHandler);
        $_SESSION = array();
        return mysql_affected_rows($this->_dbHandler) > 0;
    }
}


$sessionHandler = new sessionHandler('localhost', 'root', '123abc+');
session_set_save_handler(
                            array($sessionHandler, 'open'),
                            array($sessionHandler, 'close'),
                            array($sessionHandler, 'read'),
                            array($sessionHandler, 'write'),
                            array($sessionHandler, 'destroy'),
                            array($sessionHandler, 'gc')
                        );


/*
    在 PHP 5.0.5 中,在對象銷燬以後纔會調用 write 和 close 回調函數, 因此,在這兩個回調函數中不可使用對象,也不能夠拋出異常。 
    若是在函數中拋出異常,PHP 既不會捕獲它,也不會跟蹤它, 這樣會致使程序異常終止。 
    可是對象析構函數可使用會話。
    能夠在析構函數中調用 session_write_close() 函數來解決這個問題。 
    可是註冊 shutdown 回調函數纔是更加可靠的作法
*/
register_shutdown_function('session_write_close');

session_start();
$_SESSION['test'] = 'aa';

而後了創建一個表 叫 session  ,記住先創建數據庫'mytest'奧  session表中有三個字段

id   vchar(100)  primary  sessionid的主鍵

data vchar(1000) 數據內容(序列化後的)

last_time int(10)  最後修改的時間戳

 

整完了運行下發現表裏面的內容

你們能夠看得出,經過代碼自定義session的這種方式不只能夠應用到數據庫上,也可使用其餘的,如文件、redis之類

 

 

至此,session的原理,如何自定義存放session,在集羣中如何使用session,就已經完了

相關文章
相關標籤/搜索