文章目录
- 一、为什么要引入 session 共享
- 二、配置 Redis 实现 session 共享
一、为什么要引入 session 共享
参考雨哥的文章:http://springboot.javaboy.org/2019/0604/springboot-springsession
在传统的单服务架构中,一般来说,只有一个服务器,那么不存在 Session 共享问题,但是在分布式/集群项目中,Session 共享则是一个必须面对的问题,先看一个简单的架构图:
在这样的架构中,会出现一些单服务中不存在的问题,例如客户端发起一个请求,这个请求到达 Nginx 上之后,被 Nginx 转发到 Tomcat A 上,然后在 Tomcat A 上往 session 中保存了一份数据,下次又来一个请求,这个请求被转发到 Tomcat B 上,此时再去 Session 中获取数据,发现没有之前的数据。
对于这一类问题的解决,思路很简单,就是将各个服务之间需要共享的数据,保存到一个公共的地方(主流方案就是 Redis):
当所有 Tomcat 需要往 Session 中写数据时,都往 Redis 中写,当所有 Tomcat 需要读数据时,都从 Redis 中读。这样,不同的服务就可以使用相同的 Session 数据了。
这样的方案,可以由开发者手动实现,即手动往 Redis 中存储数据,手动从 Redis 中读取数据,相当于使用一些 Redis 客户端工具来实现这样的功能,毫无疑问,手动实现工作量还是蛮大的。
一个简化的方案就是使用 Spring Session 来实现这一功能,Spring Session 就是使用 Spring 中的代理过滤器,将所有的 Session 操作拦截下来,自动的将数据 同步到 Redis 中,或者自动的从 Redis 中读取数据。
对于开发者来说,所有关于 Session 同步的操作都是透明的,开发者使用 Spring Session,一旦配置完成后,具体的用法就像使用一个普通的 Session 一样。
二、配置 Redis 实现 session 共享
关于 redis的配置可参考:一篇文章带你搞定 SpringBoot 整合 Redis
(1)配置 redis
#配置 redis
spring.redis.host=192.168.245.131
spring.redis.database=0
spring.redis.port=6379
#我这里未设置密码,所以默认为空
#spring.redis.password=
#初始设置端口为 8080
server.port=8080
(2)编写测试代码 HelloController
@RestController
public class HelloController {
@Value("${server.port}")
Integer port;
@GetMapping("/set")
public String set(HttpSession session) {
session.setAttribute("name", "Yolo");
return String.valueOf(port);
}
@GetMapping("/get")
public String get(HttpSession session) {
Object name = session.getAttribute("name");
return (String) name + ":" + port;
}
}
这里测试 8080 端口设置session ,然后通过 8082端口访问,看看是否能够访问到 session的值
登录:
在 8080 端口设置 session 成功:
然后将这个项目打包,再通过 8081 端口访问:
然后通过 终端在 8082 端口启动该项目测试session的读取:
读取成功,说明 session 实现了共享:
这里的 session 共享是手动切换服务实例,接下来引入 Nginx ,实现服务的自动切换
使用 nginx自动化配置可参考:一篇文章带你搞定 SpringBoot 中 使用 nginx 实现 session 共享