利用Spring Session和redis对Session进行共享详解
作者:a60782885 发布时间:2021-10-18 18:22:45
前言
我们在搭建完集群环境后,不得不考虑的一个问题就是用户访问产生的session如何处理。
session的处理有很多种方法,详情见转载的上篇博客:集群/分布式环境下5种session处理策略
在这里我们讨论其中的第三种方法:session共享。
redis集群做主从复制,利用redis数据库的最终一致性,将session信息存入redis中。当应用服务器发现session不在本机内存的时候,就去redis数据库中查找,因为redis数据库是独立于应用服务器的数据库,所以可以做到session的共享和高可用。
不足:
1.redis需要内存较大,否则会出现用户session从Cache中被清除。
2.需要定期的刷新缓存
初步结构如下:
但是这个结构仍然存在问题,redis master是一个重要瓶颈,如果master崩溃的时候,但是redis不会主动的进行master切换,这时session服务中断。
但是我们先做到这个结构,后面再进行优化修改。
Spring Boot提供了Spring Session来完成session共享。
官方文档:http://docs.spring.io/spring-session/docs/current/reference/html5/guides/boot.html#boot-sample
首先创建简单的Controller:
@Controller
public class UserController {
@RequestMapping(value="/main", method=RequestMethod.GET)
public String main(HttpServletRequest request) {
HttpSession session = request.getSession();
String sessionId = (String) session.getAttribute("sessionId");
if (null != sessionId) { // sessionId不为空
System.out.println("main sessionId:" + sessionId);
return "main";
} else { // sessionId为空
return "redirect:/login";
}
}
@RequestMapping(value="/login", method=RequestMethod.GET)
public String login() {
return "login";
}
@RequestMapping(value="/doLogin", method=RequestMethod.POST)
public String doLogin(HttpServletRequest request) {
System.out.println("I do real login here");
HttpSession session = request.getSession();
String sessionId = UUID.randomUUID().toString();
session.setAttribute("sessionId", sessionId);
System.out.println("login sessionId:" + sessionId);
return "redirect:/main";
}
}
简单来说就是模拟一下权限控制,如果sessionId存在就访问主页,否则就跳转到登录页面。
那么如何实现session共享呢?
加入以下依赖:
<!-- spring session -->
<dependency>
<groupId>org.springframework.session</groupId>
<artifactId>spring-session</artifactId>
<version>1.3.0.RELEASE</version>
</dependency>
<!-- redis -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-redis</artifactId>
</dependency>
增加配置类:
@EnableRedisHttpSession
public class HttpSessionConfig {
@Bean
public JedisConnectionFactory connectionFactory() {
return new JedisConnectionFactory();
}
}
这个配置类有什么用呢?
官方文档:
The Spring configuration is responsible for creating a Servlet Filter that replaces the HttpSession implementation with an implementation backed by Spring Session.
也就是说,这个配置类可以创建一个过滤器,这个过滤器支持Spring Session代替HttpSession发挥作用。
The @EnableRedisHttpSession annotation creates a Spring Bean with the name of springSessionRepositoryFilter that implements Filter. The filter is what is in charge of replacing the HttpSession implementation to be backed by Spring Session. In this instance Spring Session is backed by Redis.
@EnableRedisHttpSession注解会创建一个springSessionRepositoryFilter的bean对象去实现这个过滤器。过滤器负责代替HttpSession。
也就是说,HttpSession不再发挥作用,而是通过过滤器使用redis直接操作Session。
在application.properties中添加redis的配置:
spring.redis.host=localhost
spring.redis.password=
spring.redis.port=6379
这样,就完成了Session共享了。是不是很简单?业务代码甚至不需要一点点的修改。
验证:
一开始redis数据库是空的。
运行项目:
访问页面之后,可以在redis中看到session的信息。
随便登陆之后:
进入到了main中。说明当前这个session中是存在sessionId的。
我们查看当前页面的cookie。也就是说,这个cookie是存在sessionId的。
再运行一个新的项目,端口为8081。在原本的浏览器中直接打开一个新的标签页,我们知道,这个时候cookie是共享的。访问localhost:8081/main
我们直接访问新的项目成功了!!同一个cookie,可以做到session在不同web服务器中的共享。
最后再次强调:
HttpSession的实现被Spring Session替换,操作HttpSession等同于操作redis中的数据。
来源:http://blog.csdn.net/a60782885/article/details/70244305


猜你喜欢
- 本文实例讲述了Android控件之TabHost用法。分享给大家供大家参考。具体如下:以下通过TabHost实现android选项卡。mai
- package com.test; import java.io.FileNotFoundException;&nbs
- hystrixDashboard服务监控除了隔离依赖服务的调用以外,Hystrix还提供了准实时的调用监控(Hystrix Dashboar
- 类和结构是.NET Framework中的同样类型系统的两种基本构造。两者在本质上都属于数据结构,封装这一组整体作为一个逻辑单位的数据和行为
- Required String parameter xxx is not present类型异常异常报错学习Spring Boot的时候做一
- 本文实例讲述了java多线程下载。分享给大家供大家参考,具体如下:使用多线程下载文件可以更快完成文件的下载,多线程下载文件之所以快,是因为其
- Java 中的内部类这是一个 Java 内部类的简单实现:public class OutterJava { pr
- Android ViewGroup中的Scroller与computeScroll的有什么关系?答:没有直接的关系知道了答案,是不是意味着下
- 本文实例为大家分享了C#导出Excel的具体代码,供大家参考,具体内容如下using System;using System.Collect
- 我们都知道Android应用软件基本上都会用到登录注册功能,那么对一个一个好的登录注册模块进行封装就势在必行了。这里给大家介绍一下我的第一个
- 目录位运算按位“与” &按位“或” |异或 ^移位运算左移 <<右移 >>无符号右移 >>>
- java项目中常用maven工具来进行工程管理,但经常遇到的一个问题是生成的jar包越来越大,编译一次工程越来越慢。怎么有效地去除冗余依赖,
- 在程序中,进行类型转换是常见的事,C#支持基本的强制类型转换方法,例如:Object obj1 = new NewType();NewTyp
- 为什么写?今天去上班的公交上,有朋友在张队(张善友)的微信群里,发了一个介绍C# 6.0新特性的视频,视频7分钟,加上本人英语实在太low,
- Java-JDK * (AOP)使用及实现原理分析第一章:代理的介绍介绍:我们需要掌握的程度 * (理解) 基于反射机制掌握的程度:1.
- 我们在使用SpringData JPA框架时,进行条件查询,如果是固定条件的查询,我们可以使用符合框架规则的自定义方法以及@Query注解实
- Java数字格式类以下两个类可用于格式化和解析数字:java.text.NumberFormatjava.text.DecimalForma
- 适用场合: 7.3工厂模式的适用场合 创建新对象最简单的办法是使用new关键字和具体类。只有在某些场合下,创建和维护对象工厂所带来的额外复杂
- 一、概述热修复这项技术,基本上已经成为项目比较重要的模块了。主要因为项目在上线之后,都难免会有各种问题,而依靠发版去修复问题,成本太高了。现
- 为了保持类型的安全性,默认情况下 C# 是不支持指针的,但是如果使用 unsafe 关键字来修饰类或类中的成员,这样的类或类中成员就会被视为