负载均衡中使用 Redis 实现共享 Session

最近在研究Web架构方面的知识,包括数据库读写分离,Redis缓存和队列,集群,以及负载均衡(LVS),今天就来先学习下我在负载均衡中遇到的问题,那就是session共享的问题。

一、负载均衡

负载均衡:把众多的访问量分担到其他的服务器上,让每个服务器的压力减少。

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

如我们第一次访问 www.baidu.com 这个域名,可能会对应这个IP111.13.101.208的服务器,然后第二次访问,IP可能会变为 111.13.101.209的服务器,这就是百度采用了负载均衡,一个域名对应多个服务器,将访问量分担到其他的服务器,这样很大程度的减轻了每个服务器上访问量。

640?wx_fmt=png

但是,这里有一个问题,如果我们登录了百度的一个账号,如网页的百度网盘,但是每次有可能请求的是不同的服务器,我们知道每个服务器都会有自己的会话session,所以会导致用户每次刷新网页又要重新登录,这是非常糟糕的体验,因此,根据以上问题,希望session可以共享,这样就可以解决负载均衡中同一个域名不同服务器对应不同session的问题。

二、Redis介绍

目前多服务器的共享session,用的最多的是redis。

关于Redis的基础知识,可以看我之前的博文Redis开发学习。

再简单的梳理下:

1.redis是key-value的存储系统,属于非关系型数据库

2.特点:支持数据持久化,可以让数据在内存中保存到磁盘里(memcached:数据存在内存里,如果服务重启,数据会丢失)

3.支持5种数据类型:string,hash,list,set,zset

4.两种文件格式(即数据持久化)

  1. RDB(全量数据):多长时间/频率,把内存中的数据刷到磁盘中,便于下次读取文件时进行加载。

  2. AOF(增量请求):类似mysql的二进制日志,不停地把对数据库的更改语句记录到日志中,下次重启服务,会根据二进制日志把数据重写一次,加载到内存里,实现数据持久化

5.存储

  1. 内存存储

  2. 磁盘存储(RDB)

  3. 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。

0?wx_fmt=png

这里有三种修改方式:

1.修改配置文件php.ini

找到配置文件 php.ini,修改为下面内容,保存并重启服务

 
  1. session.save_handler = redis

  2. session.save_path = "tcp://127.0.0.1:6379"

2.代码中动态配置修改

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

 
  1. ini_set("session.save_handler", "redis");

  2. 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的时候会报错。

测试:

 
  1. <?php

  2. //ini_set("session.save_handler", "redis");

  3. //ini_set("session.save_path", "tcp://127.0.0.1:6379");

  4.  

  5. session_start();

  6.  

  7. //存入session

  8. $_SESSION['class'] = array('name' => 'toefl', 'num' => 8);

  9.  

  10. //连接redis

  11. $redis = new redis();

  12. $redis->connect('127.0.0.1', 6379);

  13.  

  14. //检查session_id

  15. echo 'session_id:' . session_id() . '<br/>';

  16.  

  17. //redis存入的session(redis用session_id作为key,以string的形式存储)

  18. echo 'redis_session:' . $redis->get('PHPREDIS_SESSION:' . session_id()) . '<br/>';

  19.  

  20. //php获取session值

  21. echo 'php_session:' . json_encode($_SESSION['class']);

3.自定义会话机制

使用 session_set_save_handle 方法自定义会话机制,网上发现了一个封装非常好的类,我们可以直接使用这个类来实现我们的共享session操作。

 
  1. <?php

  2. class redisSession{

  3.    /**

  4.     * 保存session的数据库表的信息

  5.     */

  6.    private $_options = array(

  7.        'handler' => null, //数据库连接句柄

  8.        'host' => null,

  9.        'port' => null,

  10.        'lifeTime' => null,

  11.        'prefix'   => 'PHPREDIS_SESSION:'

  12.    );

  13.  

  14.    /**

  15.     * 构造函数

  16.     * @param $options 设置信息数组

  17.     */

  18.    public function __construct($options=array()){

  19.        if(!class_exists("redis", false)){

  20.            die("必须安装redis扩展");

  21.        }

  22.        if(!isset($options['lifeTime']) || $options['lifeTime'] <= 0){

  23.            $options['lifeTime'] = ini_get('session.gc_maxlifetime');

  24.        }

  25.        $this->_options = array_merge($this->_options, $options);

  26.    }

  27.  

  28.    /**

  29.     * 开始使用该驱动的session

  30.     */

  31.    public function begin(){

  32.        if($this->_options['host'] === null ||

  33.           $this->_options['port'] === null ||

  34.           $this->_options['lifeTime'] === null

  35.        ){

  36.            return false;

  37.        }

  38.        //设置session处理函数

  39.        session_set_save_handler(

  40.            array($this, 'open'),

  41.            array($this, 'close'),

  42.            array($this, 'read'),

  43.            array($this, 'write'),

  44.            array($this, 'destory'),

  45.            array($this, 'gc')

  46.        );

  47.    }

  48.    /**

  49.     * 自动开始回话或者session_start()开始回话后第一个调用的函数

  50.     * 类似于构造函数的作用

  51.     * @param $savePath 默认的保存路径

  52.     * @param $sessionName 默认的参数名,PHPSESSID

  53.     */

  54.    public function open($savePath, $sessionName){

  55.        if(is_resource($this->_options['handler'])) return true;

  56.        //连接redis

  57.        $redisHandle = new Redis();

  58.        $redisHandle->connect($this->_options['host'], $this->_options['port']);

  59.        if(!$redisHandle){

  60.            return false;

  61.        }

  62.  

  63.        $this->_options['handler'] = $redisHandle;

  64. //        $this->gc(null);

  65.        return true;

  66.  

  67.    }

  68.  

  69.    /**

  70.     * 类似于析构函数,在write之后调用或者session_write_close()函数之后调用

  71.     */

  72.    public function close(){

  73.        return $this->_options['handler']->close();

  74.    }

  75.  

  76.    /**

  77.     * 读取session信息

  78.     * @param $sessionId 通过该Id唯一确定对应的session数据

  79.     * @return session信息/空串

  80.     */

  81.    public function read($sessionId){

  82.        $sessionId = $this->_options['prefix'].$sessionId;

  83.        return $this->_options['handler']->get($sessionId);

  84.    }

  85.  

  86.    /**

  87.     * 写入或者修改session数据

  88.     * @param $sessionId 要写入数据的session对应的id

  89.     * @param $sessionData 要写入的数据,已经序列化过了

  90.     */

  91.    public function write($sessionId, $sessionData){

  92.        $sessionId = $this->_options['prefix'].$sessionId;

  93.        return $this->_options['handler']->setex($sessionId, $this->_options['lifeTime'], $sessionData);

  94.    }

  95.  

  96.    /**

  97.     * 主动销毁session会话

  98.     * @param $sessionId 要销毁的会话的唯一id

  99.     */

  100.    public function destory($sessionId){

  101.        $sessionId = $this->_options['prefix'].$sessionId;

  102. //        $array = $this->print_stack_trace();

  103. //        log::write($array);

  104.        return $this->_options['handler']->delete($sessionId) >= 1 ? true : false;

  105.    }

  106.  

  107.    /**

  108.     * 清理绘画中的过期数据

  109.     * @param 有效期

  110.     */

  111.    public function gc($lifeTime){

  112.        //获取所有sessionid,让过期的释放掉

  113.        //$this->_options['handler']->keys("*");

  114.        return true;

  115.    }

  116.    //打印堆栈信息

  117.    public function print_stack_trace()

  118.    {

  119.        $array = debug_backtrace ();

  120.        //截取用户信息

  121.        $var = $this->read(session_id());

  122.        $s = strpos($var, "index_dk_user|");

  123.        $e = strpos($var, "}authId|");

  124.        $user = substr($var,$s+14,$e-13);

  125.        $user = unserialize($user);

  126.        //print_r($array);//信息很齐全

  127.        unset ( $array [0] );

  128.        if(!empty($user)){

  129.          $traceInfo = $user['id'].'|'.$user['user_name'].'|'.$user['user_phone'].'|'.$user['presona_name'].'++++++++++++++++\n';

  130.        }else{

  131.          $traceInfo = '++++++++++++++++\n';

  132.        }

  133.        $time = date ( "y-m-d H:i:m" );

  134.        foreach ( $array as $t ) {

  135.            $traceInfo .= '[' . $time . '] ' . $t ['file'] . ' (' . $t ['line'] . ') ';

  136.            $traceInfo .= $t ['class'] . $t ['type'] . $t ['function'] . '(';

  137.            $traceInfo .= implode ( ', ', $t ['args'] );

  138.            $traceInfo .= ")\n";

  139.        }

  140.        $traceInfo .= '++++++++++++++++';

  141.        return $traceInfo;

  142.    }

  143.  

  144. }

在你的项目入口处调用上边的类:上边的方法等于是重写了session写入文件的方法,将数据写入到了Redis中。

初始化文件 init.php

 
  1. <?php

  2. require_once("redisSession.php");

  3. $handler = new redisSession(array(

  4.                'host' => "127.0.0.1",

  5.                'port' => "6379"

  6.        ));

  7. $handler->begin();

  8.  

  9. // 这也是必须的,打开session,必须在session_set_save_handler后面执行

  10. session_start();

测试 test.php

 
  1. <?php

  2. // 引入初始化文件

  3. include("init.php");

  4. $_SESSION['isex'] = "Hello";  

  5. $_SESSION['sex']  = "Corwien";

  6.  

  7. // 打印文件

  8. print_r($_SESSION);

  9. // ( [sex] => Corwien [isex] => Hello )

在Redis客户端使用命令查看我们的这条数据是否存在:

 
  1. 27.0.0.1:6379> keys *

  2. 1) "first_key"

  3. 2) "mylist"

  4. 3) "language"

  5. 4) "mytest"

  6. 5) "pragmmer"

  7. 6) "good"

  8. 7) "PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4"

  9. 8) "user:1"

  10. 9) "counter:__rand_int__"

  11. 10) "key:__rand_int__"

  12. 11) "tutorial-list"

  13. 12) "id:1"

  14. 13) "name"

  15. 127.0.0.1:6379> get PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4

  16. "sex|s:7:\"Corwien\";isex|s:5:\"Hello\";"

  17. 127.0.0.1:6379>

我们可以看到,我们的数据被保存在了Redis端了,键为: PHPREDIS_SESSION:29a111bcs120sv48ibmmjqdag4.

转载于:https://www.cnblogs.com/ameijiemu/p/9070392.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/414009.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Typora中使用Gitee图床

1、前言 之前好友写了一篇「使用gitee作为图床 ,写markdown自动上传文件」&#xff0c;初衷是由于我一直使用的是Typora来写博客「力推」&#xff0c;但之前的版本都不支持图床功能&#xff0c;现在新版本已经有了图床功能了&#xff0c;赶紧入坑。 本篇环境&#xff1a;MacOS…

【NOIP2017模拟6.25】小W的动漫

题目 小W最近迷上了日本动漫&#xff0c;每天都有无数部动漫的更新等着他去看&#xff0c;所以他必须将所有的动漫排个顺序&#xff0c;当然&#xff0c;虽然有无数部动漫&#xff0c;但除了1号动漫&#xff0c;每部动漫都有且仅有一部动漫是它的前传&#xff08;父亲&#xff…

用Elasticsearch代替数据库存储日志方式

之前的项目中一直使用的是数据库表记录用户操作日志的&#xff0c;但随着时间的推移&#xff0c;数据库log单表是越来越大「不考虑删除」&#xff0c;再加上近期项目中需要用到Elasticsearch&#xff0c;所以干脆把这些用户日志迁移到ES上来了。 环境&#xff1a;SpringBoot2.2…

[js] 写一个方法实现promise失败后自动重试

[js] 写一个方法实现promise失败后自动重试 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><meta http-equiv&…

如何理解Java中的自动拆箱和自动装箱?

小伟刚毕业时面的第一家公司就被面试官给问住了&#xff0c;记忆尤深啊… 如何理解Java中的自动拆箱和自动装箱&#xff1f; 自动拆箱&#xff1f;自动装箱&#xff1f;什么鬼&#xff0c;听都没听过啊&#xff0c;这…这…知识盲区… 回到家后小伟赶紧查资料&#xff0c;我…

基于Docker的Redis集群简单搭建

环境&#xff1a;Docker ( Redis:5.0.5 * 3 ) 1、拉取镜像 docker pull redis:5.0.52、创建Redis容器 创建三个 redis 容器&#xff1a; redis-node1&#xff1a;6379redis-node2&#xff1a;6380redis-node3&#xff1a;6381 docker create --name redis-node1 -v /data…

Python 全栈开发十 socket网络编程

一、客户端&#xff08;client&#xff09;服务端&#xff08;sever&#xff09;架构 在计算机中有很多常见的C/S架构&#xff0c;例如我们的浏览器是客户端、而百度网站和其他的网站就是服务端&#xff1b;视频软件是客户端&#xff0c;提供视频的腾讯、优酷、爱奇艺就是服务端…

基于Docker方式实现Elasticsearch集群

文本环境&#xff1a;Docker (Elasticsearch6.8.5 * 3) 1、拉取Elasticsearch docker pull elasticsearch6.8.52、创建es挂载目录 创建3个文件夹用于存放es挂载地址&#xff1a;es01、es02、es03 [rootCentOS7 ~]# mkdir /es-cluster [rootCentOS7 ~]# cd /es-cluster/ [ro…

基于Docker搭建Gitlab代码存储

关于Docker搭建Gitlab&#xff0c;在19年时就已经在博客发过文章了&#xff0c;今天重新回顾一下。 1、拉取镜像 docker pull gitlab/gitlab-ce默认拉取最新版本&#xff1a; 2、创建Gitlab配置 创建GitLab 的配置 (etc) 、 日志 (log) 、数据 (data) 放到容器之外&#xff…

读书笔记--Android Gradle权威指南(上)

本篇文章已授权微信公众号 dasu_Android&#xff08;大苏&#xff09;独家发布 最近看了一本书《Android Gradle 权威指南》&#xff0c;对于 Gradle 理解又更深了&#xff0c;但不想过段时间就又忘光了&#xff0c;所以打算写一篇读书笔记&#xff0c;将书中一些我个人觉得蛮有…

基于Docker搭建私有镜像仓库

通常我们在docker中拉取的镜像都是在docker hub在线存储库中获取的&#xff0c;这个在线存储库里的docker镜像可以由任何用户发布和使用&#xff0c;显然这在某些场景下是不适用的&#xff0c;比如某些互金的隐私项目&#xff0c;或者是公司完全处于内网状态不能访问外网&#…

volatile理解了吗?

到这里大家感觉自己对volatile理解了吗&#xff1f; 如果理解了&#xff0c;大家考虑这么一个问题&#xff1a;ReentrantLock&#xff08;或者其它基于AQS实现的锁&#xff09;是如何保证代码段中变量&#xff08;变量主要是指共享变量&#xff0c;存在竞争问题的变量&…

Linux|CentOS下配置Maven环境

1、下载maven包 wget http://mirrors.tuna.tsinghua.edu.cn/apache/maven/maven-3/3.3.9/binaries/apache-maven-3.3.9-bin.tar.gz如果提示 wget: 未找到命令&#xff0c;请尝试如下指令安装 wget yum -y install wget2、解压下载的maven压缩吧 tar -xzvf apache-maven-3.3.…

CentOS中安装Docker步骤

1、安装仓库所需要的软件包 yum install -y yum-utils device-mapper-persistent-data lvm22、设置yum加速源 yum-config-manager --add-repo http://mirrors.aliyun.com/repo/Centos-7.repo3、安装docker-ce yum install docker-ce docker-ce-cli containerd.io4、启动dock…

Docker+Jenkins+Git+GitLab实现DevOps

先了解一下Jenkins Jenkins是一个开源的、提供友好操作界面的持续集成(CI)工具&#xff0c;提供了数百个插件来支持构建&#xff0c;部署和自动化任何项目。我们可以使用Jenkins结合常用的版本控制工具(git、svn等)来实现自动部署项目&#xff0c;比如说我们从本地上传代码到G…

毕业两年的大专生程序员工作总结(java后端)

文章目录前言这一年做了啥去年的学习清单今年的学习清单第三年的规划最后唠叨的话前言 如题&#xff0c;这是我毕业第二年的工作总结&#xff0c;对第一年工作总结感兴趣的请戳这《毕业一年的大专生程序员工作总结》&#xff0c;再简单介绍一下我以及这个系列的文章。 关于我…

Docker开启远程安全访问

一、编辑docker.service文件 vi /usr/lib/systemd/system/docker.service找到 [Service] 节点&#xff0c;修改 ExecStart 属性&#xff0c;增加 -H tcp://0.0.0.0:2375 ExecStart/usr/bin/dockerd -H fd:// --containerd/run/containerd/containerd.sock -H tcp://0.0.0.0:23…

Netty入门篇-从双向通信开始

百度百科描述 Netty是由JBOSS提供的一个java开源框架&#xff0c;现为 Github上的独立项目。Netty提供异步的、事件驱动的网络应用程序框架和工具&#xff0c;用以快速开发高性能、高可靠性的网络服务器和客户端程序。 也就是说&#xff0c;Netty 是一个基于NIO的客户、服务器…

asp.net 页面静态化

页面静态化,有三种方式 伪静态 真静态,折中法 现在我做的是折中发 创建一个asp.net 页面, 连接跳转到还未生成的页面 创建HttpHandle类 using System;using System.Collections.Generic;using System.Linq;using System.Web;using System.IO;/// <summary>/// HttpHa…

Netty心跳机制-长连接

前文需求回顾 完成对红酒窖的室内温度采集及监控功能。由本地应用程序温度传感器定时采集室内温度上报至服务器&#xff0c;如果温度 >20 C 则由服务器下发重启空调指令&#xff0c;如果本地应用长时间不上传温度给服务器&#xff0c;则给户主手机发送一条预警短信。 Netty…