首页 » Web前端 » redissession共享php技巧_负载均衡中运用 Redis 实现共享 Session

redissession共享php技巧_负载均衡中运用 Redis 实现共享 Session

访客 2024-11-03 0

扫一扫用手机浏览

文章目录 [+]

一、负载均衡

负载均衡:把浩瀚的访问量分担到其他的做事器上,让每个做事器的压力减少。

redissession共享php技巧_负载均衡中运用 Redis 实现共享 Session

普通的阐明便是:把一项任务交由一个开拓职员处理总会有上限处理能力,这时可以考虑增加开拓职员来共同处理这项任务,多人处理同一项任务时就会涉及到调度问题,即任务分配,这和多线程理念是同等的。
nginx在这里的角色相称于任务分配者。

redissession共享php技巧_负载均衡中运用 Redis 实现共享 Session
(图片来自网络侵删)

如我们第一次访问 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\"大众

2.代码中动态配置修正

直接在代码中加入以下内容:

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']);

3.自定义会话机制

利用 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.

标签:

相关文章

介绍百度码,技术革新背后的智慧之光

随着科技的飞速发展,互联网技术已经成为我们生活中不可或缺的一部分。而在这个信息爆炸的时代,如何快速、准确地获取信息,成为了人们关注...

Web前端 2025-01-03 阅读1 评论0

介绍皮箱密码,开启神秘之门的钥匙

皮箱,作为日常生活中常见的收纳工具,承载着我们的珍贵物品。面对紧闭的皮箱,许多人却束手无策。如何才能轻松打开皮箱呢?本文将为您揭秘...

Web前端 2025-01-03 阅读1 评论0

介绍盗号器,网络安全的隐忧与应对步骤

随着互联网的快速发展,网络安全问题日益突出。盗号器作为一种非法工具,对网民的个人信息安全构成了严重威胁。本文将深入剖析盗号器的原理...

Web前端 2025-01-03 阅读1 评论0