使用Spring Session redis进行Session共享

本文探讨了在集群环境中使用Spring Session结合Redis进行Session共享的解决方案。通过配置Spring Boot应用,利用@EnableRedisHttpSession注解创建过滤器,替代HttpSession以Redis为存储后端,实现了跨应用服务器的Session共享。配置包括在application.properties中设置Redis连接信息,并通过简单Controller演示了其工作原理。这种方法简化了业务代码,无需修改即可实现Session的高可用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在搭建完集群环境后,不得不考虑的一个问题就是用户访问产生的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>

增加配置类:

package com.cetc.uavsystem.redis.session;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.session.data.redis.config.annotation.web.http.EnableRedisHttpSession;

/**
 * 实现通过Redis Session共享
 * @author Mickle_zhang
 *
 */
@EnableRedisHttpSession
public class HttpSessionConfig {

	@Value("${spring.redis.host}")
	private String host;
	@Value("${spring.redis.port}")
	private int port;
	@Value("${spring.redis.database}")
	private int database;
	@Value("${spring.redis.password}")
	private String password;
	@Value("${spring.redis.timeout}")
	private int timeout;

	@Bean(name = "jedisConnectionFactory")
	public JedisConnectionFactory connectionFactory() {
		JedisConnectionFactory factory = new JedisConnectionFactory();
		factory.setHostName(host);
		factory.setPort(port);
		factory.setDatabase(database);
		factory.setPassword(password);
		factory.setTimeout(timeout);
		return factory;
	}

}

这个配置类有什么用呢?
官方文档:

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的信息

 

 

我们直接访问新的项目成功了!!同一个cookie,可以做到session在不同web服务器中的共享。

 

最后再次强调:

HttpSession的实现被Spring Session替换,操作HttpSession等同于操作redis中的数据。

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值