系列六、Springboot整合Spring Session

一、概述

在互联网发展的起始阶段,一般使用的是单服务架构,由于只有一台服务器(Tomcat),所有的请求和响应都是基于这台服务器实现的,那么就不存在session共享的问题,但是在互联网发展的今天,基本上是分布式 + 微服务了,再使用传统的方案去处理session,显然是行不通的,先看一组简单的架构图:

在上述的架构中,会出现一些单服务中不存在的问题,例如客户端发送一个请求,经Nginx转发后请求落到Tomcat1上了,然后在Tomcat1的session中保存了一份数据,下次又来一个请求,经过Nginx转发之后请求落到Tomcat2上了,那么这个时候对于第二个请求来说,想从session中拿数据显示是无法拿到的。针对这种情况,Springboot也提供了对应的解决方案,即:Spring Session。所谓Spring Session其实也不是啥新鲜玩意儿,其实就是一个HttpSession,通过引入 spring-session-data-redis 和 spring-boot-starter-data-redis,以后所有关于session的操作,Spring Session都将使用Spring中的代理过滤器,将所有的Session操作拦截下来,自动地将Session中的数据同步到Redis中去,或者从Redis中取出数据然后响应给客户端,这对客户端是无感的,存数据不会再具体的存储到某一台服务器了,而是存到一个公共的地方,取数据也从公共的地方去取,这样就解决了微服务+分布式架构下session共享的问题,架构图如下:

二、案例代码

2.1、环境搭建

2.1.1、项目概览

2.1.2、pom

XML 复制代码
<dependencies>
	<!-- springboot -->
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-web</artifactId>
	</dependency>
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-devtools</artifactId>
		<scope>runtime</scope>
		<optional>true</optional>
	</dependency>
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-test</artifactId>
		<scope>test</scope>
	</dependency>
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-data-redis</artifactId>
	</dependency>
	<dependency>
		<groupId>org.springframework.session</groupId>
		<artifactId>spring-session-data-redis</artifactId>
	</dependency>

	<!-- 工具 -->
	<dependency>
		<groupId>org.projectlombok</groupId>
		<artifactId>lombok</artifactId>
		<optional>true</optional>
		<version>1.18.30</version>
	</dependency>
	<dependency>
		<groupId>org.apache.commons</groupId>
		<artifactId>commons-collections4</artifactId>
		<version>4.1</version>
	</dependency>
	<dependency>
		<groupId>org.apache.commons</groupId>
		<artifactId>commons-lang3</artifactId>
	</dependency>
	<dependency>
		<groupId>org.apache.commons</groupId>
		<artifactId>commons-pool2</artifactId>
	</dependency>
	<dependency>
		<groupId>com.alibaba</groupId>
		<artifactId>fastjson</artifactId>
		<version>1.2.76</version>
	</dependency>
	<dependency>
		<groupId>com.alibaba.fastjson2</groupId>
		<artifactId>fastjson2</artifactId>
		<version>2.0.37</version>
	</dependency>
	<dependency>
		<groupId>com.fasterxml.jackson.core</groupId>
		<artifactId>jackson-databind</artifactId>
	</dependency>
	<dependency>
		<groupId>com.google.guava</groupId>
		<artifactId>guava</artifactId>
		<version>20.0</version>
	</dependency>
	<dependency>
		<groupId>com.alibaba.citrus</groupId>
		<artifactId>citrus-springext-all</artifactId>
		<version>3.2.4</version>
	</dependency>
	<dependency>
		<groupId>cn.hutool</groupId>
		<artifactId>hutool-all</artifactId>
		<version>5.7.22</version>
	</dependency>
</dependencies>

2.1.3、yml

bash 复制代码
spring:
  redis:
    host: 192.168.173.232
    port: 6379
    password: 123456
    database: 0

server:
  port: 8080

2.1.4、主启动

java 复制代码
/**
 * @Author : 一叶浮萍归大海
 * @Date: 2023/12/12 15:08
 * @Description:
 */
@SpringBootApplication
public class SpringbootSessionApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringbootSessionApplication.class, args);
    }

}

2.1.5、MyRedisConfig

java 复制代码
/**
 * @Author : 一叶浮萍归大海
 * @Date: 2023/12/12 15:08
 * @Description:
 */
@Configuration
public class MyRedisConfig extends CachingConfigurerSupport {

    /**
     * 解决RedisTemplate、StringRedisTemplate中文乱码问题
     * @param connectionFactory
     * @return
     */
    @Bean
    public RedisTemplate<Object, Object> redisTemplate(LettuceConnectionFactory connectionFactory) {
        RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();
        redisTemplate.setConnectionFactory(connectionFactory);

        redisTemplate.setKeySerializer(RedisSerializer.string());
        redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());

        redisTemplate.setHashKeySerializer(RedisSerializer.string());
        redisTemplate.setHashValueSerializer(new GenericJackson2JsonRedisSerializer());

        redisTemplate.afterPropertiesSet();
        return redisTemplate;
    }

    /**
     * 解决Spring Session中文乱码
     * 思路:RedisHttpSessionConfiguration ===> createRedisTemplate ===>配置一个bean名为springSessionDefaultRedisSerializer的RedisSerializer
     * 参考:https://www.jianshu.com/p/dd1df913b1b2
     * @return
     */
    @Bean
    public RedisSerializer<Object> springSessionDefaultRedisSerializer() {
        GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer();
        return serializer;
    }

}

2.2、测试

2.2.1、说明

为了测试方便方便,我这边简易搞了一台Nginx+2个应用的模式来为大家演示Springboot整合Spring Session的大概流程,架构图如下:

思路:通过Nginx转发,在Tomcat1中的session中存一个数据,然后从Tomcat2的session中看看能否拿到session中的数据,进而验证!

2.2.2、修改nginx.conf配置信息并启动

如果对Nginx使用不熟悉的小伙伴,可以参考我之前写的 Nginx系列文章,这里就不再赘述啦!nginx.conf中增加内容如下:

bash 复制代码
# Spring Session共享
upstream sessionShareServer{
    server 127.0.0.1:8081 weight=1;
    server 127.0.0.1:8082 weight=1;
}
server {
    listen       8848;
    server_name  localhost;

    location / {
        proxy_pass    http://sessionShareServer;
        proxy_redirect default;
        
    }
}

配置解释:

upstream模块:

upstream:表示配置上游服务器

sessionShareServer:给上游服务器起一个名字(自定义)

server 127.0.0.1:8081: 配置的是一个个单独服务的地址,待会儿我们会将本地jar包上传至Linux服务器,然后分别以8081和8082端口启动

weight:权重,表示当请求过来时,如何分配请求

server模块:

listen 8848:监听的端口

server_name:服务的名称

location / :表示拦截所有以 http://192.168.173.232:8848/打头的请求

proxy_pass:表示请求转发的地址,地址为上边的上游服务器

proxy_redirect:表示设置当发生重定向请求时,nginx 自动修正响应头数据(默认是 Tomcat 返回重定向,此时重定向的地址是 Tomcat 的地址,我们需要将之修改使之成为 Nginx 的地址)

2.2.3、编译 & 打包 & 将打包的jar包上传至 linux的/applications目录

2.2.4、启动服务

nohup java -jar springboot2x16-session-0.0.1-SNAPSHOT.jar --server.port=8081 &

nohup java -jar springboot2x16-session-0.0.1-SNAPSHOT.jar --server.port=8082 &

指令解释:

nohup:表示当终端关闭时,Spring Boot应用不要停止运行;

&:表示让Spring Boot应用在后台启动;

2.2.5、后台启动日志

日志文件在/applications/nohup.out中,如果启动过程中报错的话,可以在这里查看原因。

2.2.6、set & get

相关推荐
Minyy111 小时前
SpringBoot程序的创建以及特点,配置文件,LogBack记录日志,配置过滤器、拦截器、全局异常
xml·java·spring boot·后端·spring·mybatis·logback
武昌库里写JAVA2 小时前
39.剖析无处不在的数据结构
java·vue.js·spring boot·课程设计·宠物管理
李白的粉8 小时前
基于springboot的在线教育系统
java·spring boot·毕业设计·课程设计·在线教育系统·源代码
小马爱打代码9 小时前
SpringBoot原生实现分布式MapReduce计算
spring boot·分布式·mapreduce
iuyou️9 小时前
Spring Boot知识点详解
java·spring boot·后端
一弓虽9 小时前
SpringBoot 学习
java·spring boot·后端·学习
来自星星的猫教授10 小时前
spring,spring boot, spring cloud三者区别
spring boot·spring·spring cloud
乌夷12 小时前
使用spring boot vue 上传mp4转码为dash并播放
vue.js·spring boot·dash
A阳俊yi13 小时前
Spring Boot日志配置
java·spring boot·后端
苹果酱056713 小时前
2020-06-23 暑期学习日更计划(机器学习入门之路(资源汇总)+概率论)
java·vue.js·spring boot·mysql·课程设计