负载均衡:把浩瀚的访问量分担到其他的做事器上,让每个做事器的压力减少。
普通的阐明便是:把一项任务交由一个开拓职员处理总会有上限处理能力,这时可以考虑增加开拓职员来共同处理这项任务,多人处理同一项任务时就会涉及到调度问题,即任务分配,这和多线程理念是同等的。nginx在这里的角色相称于任务分配者。

如我们第一次访问 www.baidu.com
这个域名,可能会对应这个IP111.13.101.208
的做事器,然后第二次访问,IP可能会变为111.13.101.209
的做事器,这便是百度采取了负载均衡,一个域名对应多个做事器,将访问量分担到其他的做事器,这样很大程度的减轻了每个做事器上访问量。
但是,这里有一个问题,如果我们登录了百度的一个账号,如网页的百度网盘,但是每次有可能要求的是不同的做事器,我们知道每个做事器都会有自己的会话session,以是会导致用户每次刷新网页又要重新登录,这是非常糟糕的体验,因此,根据以上问题,希望session可以共享,这样就可以办理负载均衡中同一个域名不同做事器对应不同session的问题。
二、Redis先容
目前多做事器的共享session,用的最多的是redis。
关于Redis的根本知识,可以看我之前的博文Redis开拓学习。
再大略的梳理下:
1.redis是key-value的存储系统,属于非关系型数据库
2.特点:支持数据持久化,可以让数据在内存中保存到磁盘里(memcached:数据存在内存里,如果做事重启,数据会丢失)
3.支持5种数据类型:string,hash,list,set,zset
4.两种文件格式(即数据持久化)
RDB(全量数据):多永劫光/频率,把内存中的数据刷到磁盘中,便于下次读取文件时进行加载。
AOF(增量要求):类似mysql的二进制日志,一直地把对数据库的变动语句记录到日志中,下次重启做事,会根据二进制日志把数据重写一次,加载到内存里,实现数据持久化
5.存储
内存存储
磁盘存储(RDB)
log文件(AOF)
三、实现的核心思想
首先要明确session和cookie的差异。浏览器端存的是cookie每次浏览器发要求到做事端是http 报文头是会自动加上你的cookie信息的。做事端拿着用户的cookie作为key去存储里找对应的value(session)。
同一域名下的网站的cookie都是一样的。以是无论几台做事器,无论要求分配到哪一台做事器上同一用户的cookie是不变的。也便是说cookie对应的session也是唯一的。
以是,这里只要担保多台业务做事器访问同一个redis做事器(或集群)就行了。
四、PHP会话session配置改为Redis
我们可以看到PHP默认的的session配置利用文件形式保存在做事器临时目录中,我们须要Redis作为保存session的驱动,以是,这里须要对配置文件进行修正,PHP的自定义会话机制改为Redis。
这里有三种修正办法:
1.修正配置文件php.ini
找到配置文件 php.ini
,修正为下面内容,保存并重启做事
session.save_handler = redis
session.save_path = \"大众tcp://127.0.0.1:6379\"大众
直接在代码中加入以下内容:
ini_set(\"大众session.save_handler\"大众, \"大众redis\"大众);
ini_set(\"大众session.save_path\公众, \"大众tcp://127.0.0.1:6379\公众);
注:如果配置文件redis.conf里设置了连接密码requirepass,save_path须要这样写tcp://127.0.0.1:6379?auth=authpwd ,否则保存session的时候会报错。
测试:
<?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的形式存储)
echo 'redis_session:' . $redis->get('PHPREDIS_SESSION:' . session_id) . '<br/>';
//php获取session值
echo 'php_session:' . json_encode($_SESSION['class']);
利用 session_set_save_handle
方法自定义会话机制,网上创造了一个封装非常好的类,我们可以直策应用这个类来实现我们的共享session操作。
<?php
class redisSession{
/
保存session的数据库表的信息
/
private $_options = array(
'handler' => , //数据库连接句柄
'host' => ,
'port' => ,
'lifeTime' => ,
'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'] === ||
$this->_options['port'] === ||
$this->_options['lifeTime'] ===
){
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;
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' => \公众127.0.0.1\公众,
'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客户端利用命令查看我们的这条数据是否存在:
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>
我们可以看到,我们的数据被保存在了Redis端了,键为: PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4
.