1.分布式会话
查看我们原来的代码,是基于tomcat容器实现的使用cookie传输sessionid的方式实现的登陆功能
//登陆 this.request.getSession().setAttribute("IS_LOGIN",true); this.request.getSession().setAttribute("LOGIN_USER",userModel);
当使用分布式架构之后,如果仍将登录态保存在tomcat容器中的session里的话,那么可能出现一个账户怎么登陆,都登陆不上去的状况(登陆态保存在不同的服务器上),那么怎么才能避免这种问题呢?
我们将登录态使用redis统一保存在后端的一个或者多个redis集群上,这样通过前端访问的时候,总是能够保证数据的一致性。
使用redis保存登录态
1.1 引入相应的jar包
<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> <version>2.0.5.RELEASE</version> </dependency>
1.2 实现相应的配置类,注入spring容器中
@Component @EnableRedisHttpSession(maxInactiveIntervalInSeconds = 3600) public class RedisConfig { }
1.3 序列化相应的Java实体类
@Data @NoArgsConstructor @AllArgsConstructor public class UserModel implements Serializable { rivate Integer id; @NotBlank(message = "用户名不能为空") private String name; @NotNull(message = "性别不能不填写") private Integer gender; @NotNull(message = "年龄不能不填写") @Min(value = 0,message = "年龄必须大于0岁") @Max(value = 150,message = "年龄必须小于150岁") private Integer age; @NotBlank(message = "手机号不能为空") private String telephone; private String registerMode; private String thirdPartyId; @NotBlank(message = "密码不能为空") private String encrptPassword; }
1.4 application.properties中配置相应参数
#配置springboot对redis的依赖 spring.redis.port=6379 spring.redis.host=127.0.0.1 spring.redis.database=10 #设置jedis连接池 spring.redis.jedis.pool.max-active=50 spring.redis.jedis.pool.min-idle=20
1.5 使用redisTemplate存储用户登陆态
//生成登陆凭证token,UUID String uuidToken = UUID.randomUUID().toString(); uuidToken = uuidToken.replace("-",""); //建立token和用户登录态之间的联系 redisTemplate.opsForValue().set(uuidToken,userModel); redisTemplate.expire(uuidToken,1, TimeUnit.HOURS);
1.6 前端同步修改localStorage
login.html
$.ajax({ type: "POST", contentType: "application/x-www-form-urlencoded", url: "http://"+g_host+"/user/login", data: { "telephone": $("#telphone").val(), "password":password }, xhrFields:{withCredentials:true}, success: function (data) { if(data.status == "success"){ alert("登陆成功"); var token = data.data; //将返回的token存入localStorage,以便后续的页面使用 window.localStorage["token"] = token; window.location.href="./listitem.html"; }else{ alert("登陆失败,原因为"+data.data.errMsg); } }, error: function (data) { alert("登陆失败,原因为"+data.responseText); } })
getitem.html
$("#createorder").on("click", function () { //下单前进行token的验证,没有下发token则不能下单 var token = window.localStorage["token"]; if (token == null) { alert("没有登陆,不能下单!"); window.location.href = "login.html"; return false; } $("#verifyDiv img").attr("src","http://"+g_host+"/order/generateverifycode?token="+token); $("#verifyDiv").show(); });
这样子,一个使用redis保存用户登陆状态的分布式会话就简单实现了。