网站频繁出现用户未登录现象?分布式Session实战带你解决

网站频繁出现用户未登录现象?分布式Session实战带你解决一 背景 1 为什么要处理 session 这个问题想必大多数朋友都知道 在搭建完集群或者分布式环境之后 如果不做任何处理的话 网站将频繁的出现用户未登录的现象 比如 集群中有 A B 两台服务器 用户第一次访问网站时 Nginx 将用户请求分发到 A 服务器

大家好,我是讯享网,很高兴认识大家。

一. 背景

1.为什么要处理session?

这个问题想必大多数朋友都知道,在搭建完集群或者分布式环境之后,如果不做任何处理的话,网站将频繁的出现用户未登录的现象。比如:集群中有A、B两台服务器,用户第一次访问网站时,Nginx将用户请求分发到A服务器,这时A服务器给用户创建了一个Session,当用户第二次访问网站时,假设Nginx将用户请求分发到了B服务器上,而这时B服务器并不存在用户的Session,所以就会出现用户未登录的情况,这对用户来说是不可忍受的。

所以我们在搭建集群/分布式环境之后,必须考虑的一个问题就是用户访问产生的session如何处理,即session的共享机制

2.解决方案

我们将处理Session的方式大致分为三种:

  • Session保持(也有人叫黏性Session)
  • Session复制
  • Session共享(SpringSession集成Redis)

二. Spring-Session

spring-seesion可以解决分布式 session 的共享问题。

1.介绍

  • Spring Session 是Spring的项目之一,GitHub地址:https:// github.com/spring-pro。
  • Spring Session 提供了一套创建和管理 Servlet HttpSession 的完美方案。

2.功能

spring Session 提供了 API 和实现,用于管理用户的 Session 信息。除此之外,它还提供了如下特性:

  • 将 session 所保存的状态卸载到特定的外部 session 存储汇总,如 Redis 中,他们能够以独立于应用服务器的方式提供高质量的集群。
  • 控制 sessionid 如何在客户端和服务器之间进行交换,这样的话就能很容易地编写 Restful API ,因为它可以从 HTTP 头信息中获取 sessionid ,而不必再依赖于 cookie。
  • 在非 Web 请求的处理代码中,能够访问 session 数据,比如在 JMS 消息的处理代码中。
  • 支持每个浏览器上使用多个 session,从而能够很容易地构建更加丰富的终端用户体验。
  • 当用户使用 WebSocket 发送请求的时候,能够保持 HttpSession 处于活跃状态。

3.依赖

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

讯享网

4.入口类或Config

讯享网/  * @EnableRedisHttpSession注解,开启redis集中式session管理,把所有的session都存放到了redis中  * maxInactiveIntervalInSeconds: 设置 Session 失效时间  * 使用 Redis Session 之后,原 Spring Boot 中的 server.session.timeout 属性不再生效。  */ @Configuration @EnableRedisHttpSession(maxInactiveIntervalInSeconds = 1000*60) public class RedisSessionConfig  { }

5.测试


讯享网

@RestController public class RedisSessionTest {  //登录  @RequestMapping("/login")  public String login(@RequestParam("uname") String uname,                      @RequestParam("upass") String upass,                      HttpServletRequest request){      String msg="logon failure!";      if (uname!=null && "admin".equals(uname) && "123".equals(upass)){          User u=new User();          u.setUname(uname);  u.setUpass(upass);          request.getSession().setAttribute("user",u);          msg="login successful!";      }      return msg;  }  @Value("${server.port}")  private String port;  //取Session  @GetMapping("getSession")  public Map<String,Object> getSession(HttpSession session){      Map<String,Object> map=new HashMap<>();      map.put("user",session.getAttribute("user"));      map.put("port",port);      return map;  } }

三. 小结

分别启动两个微服务项目。发现测试运行。

springboot配置session共享是非常方便的,只需要EnableRedisHttpSession注解即可。

1.解读

  • RedisSession 在创建时设置 3 个变量 creationTime ,maxInactiveInterval ,lastAccessedTime 。
  • maxInactiveInterval 默认值为 1800秒 ,表示 1800s 之内该 session 没有被再次使用,则表明该 session 已过期。
  • 每次 session 被访问都会更新 lastAccessedTime 的值, session 的过期计算公式:当前时间-lastAccessedTime > maxInactiveIn terval.

获取 session

spring session在 redis 里面保存的数据包括:

  • SET 类型的

spring:session:expireations:[min]

min 表示从 1970 年 1 月 1 日 0 点 0 分经过的分钟数, SET 集合的 member 为 expires:[sessionId] ,表示 members 会在 min 分钟后过期。

  • String 类型的

spring:session:sessions:expires:[sessionId]

该数据的 TTL 表示 sessionId 过期的剩余时间,即 maxInactiveInte rval。

  • Hash 类型的

spring:session:sessions:[sessionId]

session保存的数据,记录了creationTime,maxInactiveInterval,lastAccessedTime,attribute。前两个数据是用于 session 过期管理的辅助数据结构。

以上就是分布式Session解决方案,大家对于更多的分布式知识可以该问下面的开源地址。

2.分布式体系知识大全

https://www.processon.com/login?backUrl=/mindmap

扫码开启学习

全套Java免费资源获取

小讯
上一篇 2025-01-07 21:42
下一篇 2025-02-16 22:36

相关推荐

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请联系我们,一经查实,本站将立刻删除。
如需转载请保留出处:https://51itzy.com/kjqy/61204.html