分布式Session的几种解决方案

前言

Session应用在Web应用程序中,用于存储用户的会话信息,最常见的是保存登录信息,如果应用是单机部署,那session就会被存储在应用服务器中,而对于分布式系统来说,是如何处理的呢,这篇文章会对相关知识简单分析。

  1. session在单体应用中的使用
  2. 分布式session的解决方案

单体应用中的session

如果我们只部署一个单体应用服务,也就只存在一个Web服务器,因此所有的Session都存储在一起,所以此时的session是非常容易管理的,下面是在SpringBoot中来操作Session的简单示例。

typescript 复制代码
//设置session
@GetMapping("setSession")
public void setSession(HttpServletRequest request){
    request.getSession().setAttribute("name","hg");
}
//获取session
@GetMapping("getSession")
public String getSession(HttpServletRequest request){
    return (String) request.getSession().getAttribute("name");
}

分布式session

对于分布式系统而言,因为部署了多台服务,不同的服务维护的session内容是不一致的,比如在部署时,同一套订单系统部署了两台服务,用户在请求时随机请求其中一台服务,如果直接使用上面的方式获取session,就会出现部分请求无法获取session的情况。

这时就要考虑分布式session的解决方法,目前常见的解决方案有Session复制、Session粘带、集中性Session管理等,本文只介绍比较常用的集中性Session管理这种方案的代码示例。

  1. Session同步

通过修改tomcat的配置,让所有tomcat之间的session进行同步,这样就可以实现不同Web容器的能拿到相同的session,这种最明显的缺点就是每台tomcat服务器都保存一份全量数据,浪费服务器内存,并且会增加数据传输的开销。

  1. Session粘带

这种方式是将session依旧保存在每一台服务器上,然后通过nginx负载均衡将相同ip请求转发到同一台服务器上,当然也可以根据业务id,将相同的业务id每次都访问同一台服务器,这样就不会和代码产生耦合,需要注意的是如果入口ip是同一个则就会变成单点系统了,所以需要根据实际情况选择。

此外,这种方式存储的session依旧是存储在web服务器上,如果服务器重启对客户端用户会有影响;如果增加或减少机器,需要重新计算哈希,也会对客户端请求产生影响,比如之前业务id是100,有2台机器,根据余数判断该id请求哪台服务器,如果增加了一台服务器,就会导致原本id=100应该请求服务器2的,现在请求到服务器1上了,就会获取数据失败。

  1. 集中性Session管理

这种方式就是将session存储在第三方数据库中,比如mysql或者redis中,下面看一下由Spring Cloud提供的基于Redis的Session管理方案。

3.1引入依赖

xml 复制代码
<dependency>
    <groupId>org.springframework.session</groupId>
    <artifactId>spring-session-data-redis</artifactId>
</dependency>
<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
</dependency>

3.2 增加配置

yaml 复制代码
#端口
server:
  port: 9999
​
#redis配置
spring:
  redis:
    database: 0
    host: localhost
    port: 6379
​
​

3.3启动类增加配置

less 复制代码
//设置session过期时间60秒,默认值1800秒
@EnableRedisHttpSession(maxInactiveIntervalInSeconds = 60)
@SpringBootApplication
public class DemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }
}

3.4测试类

typescript 复制代码
    @GetMapping("setSession")
    public void setSession(HttpServletRequest request){
        request.getSession().setAttribute("name","hg");
    }
    @GetMapping("getSession")
    public String getSession(HttpServletRequest request){
        return (String) request.getSession().getAttribute("name");
    }

3.5查看redis上的数据

相关推荐
计算机毕设指导62 分钟前
基于 SpringBoot 的作业管理系统【附源码】
java·vue.js·spring boot·后端·mysql·spring·intellij-idea
Gu Gu Study3 分钟前
枚举与lambda表达式,枚举实现单例模式为什么是安全的,lambda表达式与函数式接口的小九九~
java·开发语言
Chris _data6 分钟前
二叉树oj题解析
java·数据结构
牙牙70511 分钟前
Centos7安装Jenkins脚本一键部署
java·servlet·jenkins
paopaokaka_luck19 分钟前
[371]基于springboot的高校实习管理系统
java·spring boot·后端
以后不吃煲仔饭32 分钟前
Java基础夯实——2.7 线程上下文切换
java·开发语言
进阶的架构师32 分钟前
2024年Java面试题及答案整理(1000+面试题附答案解析)
java·开发语言
The_Ticker38 分钟前
CFD平台如何接入实时行情源
java·大数据·数据库·人工智能·算法·区块链·软件工程
大数据编程之光1 小时前
Flink Standalone集群模式安装部署全攻略
java·大数据·开发语言·面试·flink
爪哇学长1 小时前
双指针算法详解:原理、应用场景及代码示例
java·数据结构·算法