美文网首页SpringBoot
Springboot和Spring Session实现sessi

Springboot和Spring Session实现sessi

作者: 意识流丶 | 来源:发表于2017-10-19 11:14 被阅读113次

HttpSession是通过Servlet容器创建和管理的,像Tomcat/Jetty都是保存在内存中的。而如果我们把web服务器搭建成分布式的集群,然后利用LVS或Nginx做负载均衡,那么来自同一用户的Http请求将有可能被分发到两个不同的web站点中去。那么问题就来了,如何保证不同的web站点能够共享同一份session数据呢?

最简单的想法就是把session数据保存到内存以外的一个统一的地方,例如Memcached/Redis等数据库中。

那么问题又来了,如何替换掉Servlet容器创建和管理HttpSession的实现呢?

设计一个Filter,利用HttpServletRequestWrapper,实现自己的 getSession()方法,接管创建和管理Session数据的工作。spring-session就是通过这样的思路实现的。

使用Spring Session和Redis的组合来代替原有的HttpSession实现Session在不同项目之间的共享

在springboot中集成Spring Session
引入spring session相关依赖

<!-- spring session -->
<dependency>
    <groupId>org.springframework.session</groupId>
    <artifactId>spring-session-data-redis</artifactId>
</dependency>

添加 SpringSession 配置类

@Configuration
@EnableRedisHttpSession(maxInactiveIntervalInSeconds = 86400*30)
public class HttpSessionConfig {
        //不需要写任何代码
}

配置类中最关键的就是 @EnableRedisHttpSession
@EnableRedisHttpSession 注解创建了一个名为 springSessionRepositoryFilterbean,负责替换 httpSession,同时由 redis 提供缓存支持
为了做到全部替换,我们要确保Servlet容器(Tomcat)对于某个请求都使用这个Filter,这个由SpringBoot负责
(具体是这样的:@EnableRedisHttpSession注解通过Import,引入了RedisHttpSessionConfiguration配置类。该配置类通过@Bean注解,向Spring容器中注册了一个SessionRepositoryFilter(SessionRepositoryFilter的依赖关系:SessionRepositoryFilter --> SessionRepository --> RedisTemplate --> RedisConnectionFactory,有兴趣可以查看源码)
maxInactiveIntervalInSeconds:设置Session失效时间,使用Redis Session之后,原springbootserver.session.timeout属性不再生效

添加验证的接口
yml或者properties文件中可以通过server.port设置端口

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

    @GetMapping("/session")
    public Object getSession(HttpServletRequest request){
        Map<String, Object> map = new HashMap<String, Object>();
        map.put("SessionId", request.getSession().getId());
        map.put("ServerPort", "服务端口号为 "+port);
        return map;
    }

访问 http://localhost:8080/session

Paste_Image.png
我们看下redis缓存的数据
Paste_Image.png
可以发现sessionId已经缓存在redis数据库中
下面我们换个端口再访问一次看看
这次我把端口换成了8888 访问:http://localhost:8888/session
Paste_Image.png
刷新了redis数据库,缓存的数据也没变
Paste_Image.png
结果中的SessionId是一致的,却是由两个不同项目工程来提供服务。这样子,SpringSession 利用拦截器 Filter 帮我们在每个请求前进行了同步设置,达到了分布式系统中 session 共享。

相关文章

网友评论

  • 还是磊哥:请问,按照文章中所讲的方式,配置后。2个请求得到的 sessionid不同。http://localhost:8080/test/session 对应 {"SessionId":"c5f8b1c2-c880-4b20-a9a9-46438d2fcaa2","ServerPort":"服务端口号为 8080"};但是 http://localhost:9090/test/session 对应 {"SessionId":"d51e9db8-2813-4e51-95d1-240a5e661d7d","ServerPort":"服务端口号为 9090"}
    还是磊哥:@意识流丶 redis是配置了的。但是除了HttpSessionConfig,我这边还有一个类@Configuration
    public class RedisConfig {
    //添加一个自己的RedisTemplate BEAN,替换默认的BEAN
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {。一会我重建一个新的项目,再试试看。
    意识流丶:主要代码和我的都一样吗,redis已经配合了吧

本文标题:Springboot和Spring Session实现sessi

本文链接:https://www.haomeiwen.com/subject/fnxuuxtx.html