php負載中使用redis實現session會話保持
同一域名下的網站的cookie都是一樣的。所以無論幾臺服務器,無論請求分配到哪一臺服務器上同一用戶的cookie是不變的。也就是說cookie對應的session也是唯一的。
所以,這裏只要保證多臺業務服務器訪問同一個redis服務器(或集群)就行了。
修改php會話緩存機制改成Redis即可,這裏有三種方式:
1,修改php的配置文件
修改php.ini文件
session.save_handler = redis
//session.save_path = "tcp://172.16.1.51:6379?auth=123123"如果redis配置的密碼需要寫成這種方式,填寫redis的密碼
session.auto_start = 1
註釋php-fpm.d/www.conf裏面的兩條內容
;php_value[session.save_handler] = files
;php_value[session.save_path] = /var/lib/php/session
更改完成之後一定要重啟php-fpm服務
2,更改對應的代碼文件
直接在對應的代碼中加上下面兩句話
ini_set("session.save_handler","redis");
ini_set("session.save_path","tcp://172.16.1.51:6379");
//如果redis設置了密碼需要改下面一句為
//ini_set("session.save_path","tcp://172.16.1.51:6379?auth=123123");
測試
<?php
//ini_set("session.save_handler", "redis");
session_start();
//存入session
$_SESSION[‘class‘] = array(‘name‘ => ‘toefl‘, ‘num‘ => 8);
//連接redis
$redis = new redis();
$redis->connect(‘127.0.0.1‘, 6379);
//檢查session_id
echo ‘session_id:‘ . session_id() . ‘<br/>‘;
//redis存入的session(redis用session_id作為key,以string的形式存儲)
echo ‘redis_session:‘ . $redis->get(‘PHPREDIS_SESSION:‘ . session_id()) . ‘<br/>‘;
//php獲取session值
echo ‘php_session:‘ . json_encode($_SESSION[‘class‘]);
3,自定義會話機制(目前不懂)
使用 session_set_save_handle 方法自定義會話機制,網上發現了一個封裝非常好的類,我們可以直接使用這個類來實現我們的共享session操作。
<?php
class redisSession{
/**
-
保存session的數據庫表的信息
*/
private $_options = array(
‘handler‘ => null, //數據庫連接句柄
‘host‘ => null,
‘port‘ => null,
‘lifeTime‘ => null,
‘prefix‘ => ‘PHPREDIS_SESSION:‘
);/**
- 構造函數
-
@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){
$sessionId = $this->_options[‘prefix‘].$sessionId;
return $this->_options[‘handler‘]->get($sessionId);
}/**
- 寫入或者修改session數據
- @param $sessionId 要寫入數據的session對應的id
-
@param $sessionData 要寫入的數據,已經序列化過了
*/
public function write($sessionId, $sessionData){
$sessionId = $this->_options[‘prefix‘].$sessionId;
return $this->_options[‘handler‘]->setex($sessionId, $this->_options[‘lifeTime‘], $sessionData);
}/**
- 主動銷毀session會話
-
@param $sessionId 要銷毀的會話的唯一id
*/
public function destory($sessionId){
$sessionId = $this->_options[‘prefix‘].$sessionId;
// $array = $this->print_stack_trace();
// log::write($array);
return $this->_options[‘handler‘]->delete($sessionId) >= 1 ? true : false;
}/**
- 清理繪畫中的過期數據
- @param 有效期
/
public function gc($lifeTime){
//獲取所有sessionid,讓過期的釋放掉
//$this->_options[‘handler‘]->keys("");
return true;
}
//打印堆棧信息
public function print_stack_trace()
{
$array = debug_backtrace ();
//截取用戶信息
$var = $this->read(session_id());
$s = strpos($var, "index_dk_user|");
$e = strpos($var, "}authId|");
$user = substr($var,$s+14,$e-13);
$user = unserialize($user);
//print_r($array);//信息很齊全
unset ( $array [0] );
if(!empty($user)){
$traceInfo = $user[‘id‘].‘|‘.$user[‘user_name‘].‘|‘.$user[‘user_phone‘].‘|‘.$user[‘presona_name‘].‘++++++++++++++++\n‘;
}else{
$traceInfo = ‘++++++++++++++++\n‘;
}
$time = date ( "y-m-d H:i:m" );
foreach ( $array as $t ) {
$traceInfo .= ‘[‘ . $time . ‘] ‘ . $t [‘file‘] . ‘ (‘ . $t [‘line‘] . ‘) ‘;
$traceInfo .= $t [‘class‘] . $t [‘type‘] . $t [‘function‘] . ‘(‘;
$traceInfo .= implode ( ‘, ‘, $t [‘args‘] );
$traceInfo .= ")\n";
}
$traceInfo .= ‘++++++++++++++++‘;
return $traceInfo;
}
}
在你的項目入口處調用上邊的類:上邊的方法等於是重寫了session寫入文件的方法,將數據寫入到了Redis中。
初始化文件 init.php
<?php
require_once("redisSession.php");
$handler = new redisSession(array(
‘host‘ => "172.16.1.51",
‘port‘ => "6379"
));
$handler->begin();
// 這也是必須的,打開session,必須在session_set_save_handler後面執行
session_start();
測試 test.php
<?php
// 引入初始化文件
include("init.php");
$_SESSION[‘isex‘] = "Hello";
$_SESSION[‘sex‘] = "Corwien";
// 打印文件
print_r($_SESSION);
// ( [sex] => Corwien [isex] => Hello )
在Redis客戶端使用命令查看我們的這條數據是否存在:
172.16.1.51:6379> keys *
1) "first_key"
2) "mylist"
3) "language"
4) "mytest"
5) "pragmmer"
6) "good"
7) "PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4"
8) "user:1"
9) "counter:__rand_int__"
10) "key:__rand_int__"
11) "tutorial-list"
12) "id:1"
13) "name"
127.0.0.1:6379> get PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4
"sex|s:7:\"Corwien\";isex|s:5:\"Hello\";"
127.0.0.1:6379>
我們可以看到,我們的數據被保存在了Redis端了,鍵為:
PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4.
php負載中使用redis實現session會話保持