负载均衡中使用 Redis 实现共享 Session
最近在研究Web架構(gòu)方面的知識(shí),包括數(shù)據(jù)庫(kù)讀寫分離,Redis緩存和隊(duì)列,集群,以及負(fù)載均衡(LVS),今天就來先學(xué)習(xí)下我在負(fù)載均衡中遇到的問題,那就是session共享的問題。
一、負(fù)載均衡
負(fù)載均衡:把眾多的訪問量分擔(dān)到其他的服務(wù)器上,讓每個(gè)服務(wù)器的壓力減少。
通俗的解釋就是:把一項(xiàng)任務(wù)交由一個(gè)開發(fā)人員處理總會(huì)有上限處理能力,這時(shí)可以考慮增加開發(fā)人員來共同處理這項(xiàng)任務(wù),多人處理同一項(xiàng)任務(wù)時(shí)就會(huì)涉及到調(diào)度問題,即任務(wù)分配,這和多線程理念是一致的。nginx在這里的角色相當(dāng)于任務(wù)分配者。
如我們第一次訪問 www.baidu.com 這個(gè)域名,可能會(huì)對(duì)應(yīng)這個(gè)IP111.13.101.208的服務(wù)器,然后第二次訪問,IP可能會(huì)變?yōu)?111.13.101.209的服務(wù)器,這就是百度采用了負(fù)載均衡,一個(gè)域名對(duì)應(yīng)多個(gè)服務(wù)器,將訪問量分擔(dān)到其他的服務(wù)器,這樣很大程度的減輕了每個(gè)服務(wù)器上訪問量。
但是,這里有一個(gè)問題,如果我們登錄了百度的一個(gè)賬號(hào),如網(wǎng)頁(yè)的百度網(wǎng)盤,但是每次有可能請(qǐng)求的是不同的服務(wù)器,我們知道每個(gè)服務(wù)器都會(huì)有自己的會(huì)話session,所以會(huì)導(dǎo)致用戶每次刷新網(wǎng)頁(yè)又要重新登錄,這是非常糟糕的體驗(yàn),因此,根據(jù)以上問題,希望session可以共享,這樣就可以解決負(fù)載均衡中同一個(gè)域名不同服務(wù)器對(duì)應(yīng)不同session的問題。
二、Redis介紹
目前多服務(wù)器的共享session,用的最多的是redis。
關(guān)于Redis的基礎(chǔ)知識(shí),可以看我之前的博文Redis開發(fā)學(xué)習(xí)。
再簡(jiǎn)單的梳理下:
1.redis是key-value的存儲(chǔ)系統(tǒng),屬于非關(guān)系型數(shù)據(jù)庫(kù)
2.特點(diǎn):支持?jǐn)?shù)據(jù)持久化,可以讓數(shù)據(jù)在內(nèi)存中保存到磁盤里(memcached:數(shù)據(jù)存在內(nèi)存里,如果服務(wù)重啟,數(shù)據(jù)會(huì)丟失)
3.支持5種數(shù)據(jù)類型:string,hash,list,set,zset
4.兩種文件格式(即數(shù)據(jù)持久化)
-
RDB(全量數(shù)據(jù)):多長(zhǎng)時(shí)間/頻率,把內(nèi)存中的數(shù)據(jù)刷到磁盤中,便于下次讀取文件時(shí)進(jìn)行加載。
-
AOF(增量請(qǐng)求):類似mysql的二進(jìn)制日志,不停地把對(duì)數(shù)據(jù)庫(kù)的更改語句記錄到日志中,下次重啟服務(wù),會(huì)根據(jù)二進(jìn)制日志把數(shù)據(jù)重寫一次,加載到內(nèi)存里,實(shí)現(xiàn)數(shù)據(jù)持久化
5.存儲(chǔ)
-
內(nèi)存存儲(chǔ)
-
磁盤存儲(chǔ)(RDB)
-
log文件(AOF)
三、實(shí)現(xiàn)的核心思想
首先要明確session和cookie的區(qū)別。瀏覽器端存的是cookie每次瀏覽器發(fā)請(qǐng)求到服務(wù)端是http 報(bào)文頭是會(huì)自動(dòng)加上你的cookie信息的。服務(wù)端拿著用戶的cookie作為key去存儲(chǔ)里找對(duì)應(yīng)的value(session)。
同一域名下的網(wǎng)站的cookie都是一樣的。所以無論幾臺(tái)服務(wù)器,無論請(qǐng)求分配到哪一臺(tái)服務(wù)器上同一用戶的cookie是不變的。也就是說cookie對(duì)應(yīng)的session也是唯一的。
所以,這里只要保證多臺(tái)業(yè)務(wù)服務(wù)器訪問同一個(gè)redis服務(wù)器(或集群)就行了。
四、PHP會(huì)話session配置改為Redis
我們可以看到PHP默認(rèn)的的session配置使用文件形式保存在服務(wù)器臨時(shí)目錄中,我們需要Redis作為保存session的驅(qū)動(dòng),所以,這里需要對(duì)配置文件進(jìn)行修改,PHP的自定義會(huì)話機(jī)制改為Redis。
這里有三種修改方式:
1.修改配置文件php.ini
找到配置文件 php.ini,修改為下面內(nèi)容,保存并重啟服務(wù)
-
session.save_handler?=?redis -
session.save_path?=?"tcp://127.0.0.1:6379"
2.代碼中動(dòng)態(tài)配置修改
直接在代碼中加入以下內(nèi)容:
-
ini_set("session.save_handler",?"redis"); -
ini_set("session.save_path",?"tcp://127.0.0.1:6379");
注:如果配置文件redis.conf里設(shè)置了連接密碼requirepass,save_path需要這樣寫tcp://127.0.0.1:6379?auth=authpwd ,否則保存session的時(shí)候會(huì)報(bào)錯(cuò)。
測(cè)試:
-
<?php -
//ini_set("session.save_handler", "redis"); -
//ini_set("session.save_path", "tcp://127.0.0.1:6379"); -
?
-
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的形式存儲(chǔ)) -
echo?'redis_session:'?.?$redis->get('PHPREDIS_SESSION:'?.?session_id())?.?'<br/>'; -
?
-
//php獲取session值 -
echo?'php_session:'?.?json_encode($_SESSION['class']);
3.自定義會(huì)話機(jī)制
使用 session_set_save_handle 方法自定義會(huì)話機(jī)制,網(wǎng)上發(fā)現(xiàn)了一個(gè)封裝非常好的類,我們可以直接使用這個(gè)類來實(shí)現(xiàn)我們的共享session操作。
-
<?php -
class?redisSession{ -
? ?/** -
? ? * 保存session的數(shù)據(jù)庫(kù)表的信息 -
? ? */ -
? ?private?$_options?=?array( -
? ? ? ?'handler'?=>?null,?//數(shù)據(jù)庫(kù)連接句柄 -
? ? ? ?'host'?=>?null, -
? ? ? ?'port'?=>?null, -
? ? ? ?'lifeTime'?=>?null, -
? ? ? ?'prefix'???=>?'PHPREDIS_SESSION:' -
? ?); -
?
-
? ?/** -
? ? * 構(gòu)造函數(shù) -
? ? * @param $options 設(shè)置信息數(shù)組 -
? ? */ -
? ?public?function?__construct($options=array()){ -
? ? ? ?if(!class_exists("redis",?false)){ -
? ? ? ? ? ?die("必須安裝redis擴(kuò)展"); -
? ? ? ?} -
? ? ? ?if(!isset($options['lifeTime'])?||?$options['lifeTime']?<=?0){ -
? ? ? ? ? ?$options['lifeTime']?=?ini_get('session.gc_maxlifetime'); -
? ? ? ?} -
? ? ? ?$this->_options?=?array_merge($this->_options,?$options); -
? ?} -
?
-
? ?/** -
? ? * 開始使用該驅(qū)動(dòng)的session -
? ? */ -
? ?public?function?begin(){ -
? ? ? ?if($this->_options['host']?===?null?|| -
? ? ? ? ? $this->_options['port']?===?null?|| -
? ? ? ? ? $this->_options['lifeTime']?===?null -
? ? ? ?){ -
? ? ? ? ? ?return?false; -
? ? ? ?} -
? ? ? ?//設(shè)置session處理函數(shù) -
? ? ? ?session_set_save_handler( -
? ? ? ? ? ?array($this,?'open'), -
? ? ? ? ? ?array($this,?'close'), -
? ? ? ? ? ?array($this,?'read'), -
? ? ? ? ? ?array($this,?'write'), -
? ? ? ? ? ?array($this,?'destory'), -
? ? ? ? ? ?array($this,?'gc') -
? ? ? ?); -
? ?} -
? ?/** -
? ? * 自動(dòng)開始回話或者session_start()開始回話后第一個(gè)調(diào)用的函數(shù) -
? ? * 類似于構(gòu)造函數(shù)的作用 -
? ? * @param $savePath 默認(rèn)的保存路徑 -
? ? * @param $sessionName 默認(rèn)的參數(shù)名,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; -
?
-
? ?} -
?
-
? ?/** -
? ? * 類似于析構(gòu)函數(shù),在write之后調(diào)用或者session_write_close()函數(shù)之后調(diào)用 -
? ? */ -
? ?public?function?close(){ -
? ? ? ?return?$this->_options['handler']->close(); -
? ?} -
?
-
? ?/** -
? ? * 讀取session信息 -
? ? * @param $sessionId 通過該Id唯一確定對(duì)應(yīng)的session數(shù)據(jù) -
? ? * @return session信息/空串 -
? ? */ -
? ?public?function?read($sessionId){ -
? ? ? ?$sessionId?=?$this->_options['prefix'].$sessionId; -
? ? ? ?return?$this->_options['handler']->get($sessionId); -
? ?} -
?
-
? ?/** -
? ? * 寫入或者修改session數(shù)據(jù) -
? ? * @param $sessionId 要寫入數(shù)據(jù)的session對(duì)應(yīng)的id -
? ? * @param $sessionData 要寫入的數(shù)據(jù),已經(jīng)序列化過了 -
? ? */ -
? ?public?function?write($sessionId,?$sessionData){ -
? ? ? ?$sessionId?=?$this->_options['prefix'].$sessionId; -
? ? ? ?return?$this->_options['handler']->setex($sessionId,?$this->_options['lifeTime'],?$sessionData); -
? ?} -
?
-
? ?/** -
? ? * 主動(dòng)銷毀session會(huì)話 -
? ? * @param $sessionId 要銷毀的會(huì)話的唯一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; -
? ?} -
?
-
? ?/** -
? ? * 清理繪畫中的過期數(shù)據(jù) -
? ? * @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; -
? ?} -
?
-
}
在你的項(xiàng)目入口處調(diào)用上邊的類:上邊的方法等于是重寫了session寫入文件的方法,將數(shù)據(jù)寫入到了Redis中。
初始化文件 init.php
-
<?php -
require_once("redisSession.php"); -
$handler?=?new?redisSession(array( -
? ? ? ? ? ? ? ?'host'?=>?"127.0.0.1", -
? ? ? ? ? ? ? ?'port'?=>?"6379" -
? ? ? ?)); -
$handler->begin(); -
?
-
// 這也是必須的,打開session,必須在session_set_save_handler后面執(zhí)行 -
session_start();
測(cè)試 test.php
-
<?php -
// 引入初始化文件 -
include("init.php"); -
$_SESSION['isex']?=?"Hello";?? -
$_SESSION['sex']??=?"Corwien"; -
?
-
// 打印文件 -
print_r($_SESSION); -
// ( [sex] => Corwien [isex] => Hello )
在Redis客戶端使用命令查看我們的這條數(shù)據(jù)是否存在:
-
27.0.0.1: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>
我們可以看到,我們的數(shù)據(jù)被保存在了Redis端了,鍵為: PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4.
轉(zhuǎn)載于:https://www.cnblogs.com/itrena/p/9070282.html
總結(jié)
以上是生活随笔為你收集整理的负载均衡中使用 Redis 实现共享 Session的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 整个瓜子脸要多少钱?
- 下一篇: MySQL Antelope和Barra