Spring Boot2.0 - 一个骚操作搞定session 共享

在这里插入图片描述

一.你需要准备什么

  • Spring Boot 2.1.5.RELEASE
  • nginx-1.12.2
  • redis-2.8.9
    本篇文章基于如上所述的版本编写并测试通过,不同的版本可能会有差异,需要大家尝试;

二.为什么需要做session共享

在传统的单服务架构中,一般来说,只有一个服务器,那么不存在 Session 共享问题,但是在分布式/集群项目中,Session 共享则是一个必须面对的问题,先看一个简单的架构图:
Spring Boot2.0 - 一个骚操作搞定session 共享_第1张图片
在这样的架构中,会出现一些单服务中不存在的问题,例如客户端发起一个请求,这个请求到达 Nginx 上之后,被 Nginx 转发到 Tomcat A 上,然后在 Tomcat A 上往 session 中保存了一份数据,下次又来一个请求,这个请求被转发到 Tomcat B 上,此时再去 Session 中获取数据,发现没有之前的数据。对于这一类问题的解决,思路很简单,就是将各个服务之间需要共享的数据,保存到一个公共的地方(主流方案就是 Redis):
Spring Boot2.0 - 一个骚操作搞定session 共享_第2张图片
当所有 Tomcat 需要往 Session 中写数据时,都往 Redis 中写,当所有 Tomcat 需要读数据时,都从 Redis 中读。这样,不同的服务就可以使用相同的 Session 数据了。

这样的方案,可以由开发者手动实现,即手动往 Redis 中存储数据,手动从 Redis 中读取数据,相当于使用一些 Redis 客户端工具来实现这样的功能,毫无疑问,手动实现工作量还是蛮大的。

一个简化的方案就是使用 Spring Session 来实现这一功能,Spring Session 就是使用 Spring 中的代理过滤器,将所有的 Session 操作拦截下来,自动的将数据 同步到 Redis 中,或者自动的从 Redis 中读取数据。

对于开发者来说,所有关于 Session 同步的操作都是透明的,开发者使用 Spring Session,一旦配置完成后,具体的用法就像使用一个普通的 Session 一样。

三.实战

3.1创建工程

首先 ,我们需要在我们的项目中新建一个module,引入 Web、Spring Session 以及 Redis:
Spring Boot2.0 - 一个骚操作搞定session 共享_第3张图片
创建之后的工程结构图如下图所示
Spring Boot2.0 - 一个骚操作搞定session 共享_第4张图片
我们来看看pom.xml文件需要引入那些依赖:

	<dependencies>
		<dependency>
			<groupId>org.springframework.bootgroupId>
			<artifactId>spring-boot-starter-data-redisartifactId>
		dependency>
		<dependency>
			<groupId>org.springframework.bootgroupId>
			<artifactId>spring-boot-starter-webartifactId>
		dependency>
		<dependency>
			<groupId>org.springframework.sessiongroupId>
			<artifactId>spring-session-data-redisartifactId>
		dependency>

		<dependency>
			<groupId>org.springframework.bootgroupId>
			<artifactId>spring-boot-starter-testartifactId>
			<scope>testscope>
		dependency>

		<dependency>
			<groupId>org.springframework.bootgroupId>
			<artifactId>spring-boot-starter-securityartifactId>
		dependency>
	dependencies>

3.2配置redis和security

spring:
  redis:
    database: 0
    host: 127.0.0.1
    password: milo
    port: 6379
  security :
    user :
       name : milo
       password : milo

3.3 代码编写

配置完成后 ,就可以使用 Spring Session 了,其实就是使用普通的 HttpSession ,其他的 Session 同步到 Redis 等操作,框架已经自动帮你完成了:

/**
 * @author: Milogenius
 * @create: 2019-06-13 08:59
 * @description:
 **/
@RestController
public class SessionController {

    @Value("${server.port}")
    Integer port;

    @GetMapping("set")
    public String set(HttpSession session){
        session.setAttribute("user","milo");
        return String.valueOf(port);
    }

    @GetMapping("get")
    public String get(HttpSession session){
        return session.getAttribute("user")+":"+ port;
    }
}

考虑到一会 Spring Boot 将以集群的方式启动 ,为了获取每一个请求到底是哪一个 Spring Boot 提供的服务,需要在每次请求时返回当前服务的端口号,因此这里我注入了 server.port 。

3.4项目打包

Spring Boot2.0 - 一个骚操作搞定session 共享_第5张图片
打包之后,启动项目的两个实例:
在这里插入图片描述

四.项目测试

然后先访问 localhost:8080/set 向 8080 这个服务的 Session 中保存一个变量,访问完成后,数据就已经自动同步到 Redis 中 了 :
在这里插入图片描述
然后,再调用 localhost:8081/get 接口,就可以获取到 8080 服务的 session 中的数据:
在这里插入图片描述
此时关于 session 共享的配置就已经全部完成了,session 共享的效果我们已经看到了,但是每次访问都是我自己手动切换服务实例,因此,接下来我们来引入 Nginx ,实现服务实例自动切换。


示例代码

  • GitHub

参考文章
1.SpringBoot一个骚操作搞定session 共享,没有比这更简单了

返回专栏目录

你可能感兴趣的:(SpringBoot)