Redis的共享session应用如何实现短信登录

网友投稿 284 2023-11-22

Redis的共享session应用如何实现短信登录

本篇内容介绍了“Redis的共享session应用如何实现短信登录”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

1. 基于 session 实现短信登录

1.1 短信登录流程1.2 实现发送短信验证码

Redis的共享session应用如何实现短信登录

前端请求说明:

说明请求方式POST请求路径/user/code请求参数phone(电话号码)返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Override     public Result sendCode(String phone, HttpSession session) {         // 1. 校验手机号         if(RegexUtils.isPhoneInvalid(phone)){             // 2. 如果不符合,返回错误信息             return Result.fail("手机号格式错误!");         }// 3. 符合,生成验证码(设置生成6位)         String code = RandomUtil.randomNumbers(6);         // 4. 保存验证码到 sessionsession.setAttribute("code", code);         // 5. 发送验证码(这里并未实现,通过日志记录)         log.debug("发送短信验证码成功,验证码:{}", code);         // 返回 ok         return Result.ok();     } }1.3 实现短信验证码登录、注册

前端请求说明

说明请求方式POST请求路径/user/login请求参数phone(电话号码);code(验证码)返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Override     public Result login(LoginFormDTO loginForm, HttpSession session) {         // 1. 校验手机号         String phone = loginForm.getPhone();         if(RegexUtils.isPhoneInvalid(phone)){             // 不一致,返回错误信息             returnResult.fail("手机号格式错误!");         }         // 2. 校验验证码         String cacheCode = (String) session.getAttribute("code");         String code = loginForm.getCode();         if(cacheCode == null|| !cacheCode.equals(cacheCode)){// 不一致,返回错误信息             return Result.fail("验证码错误!");         }         // 4. 一致,根据手机号查询用户(这里使用的 mybatis-plus)         User user = query().eq("phone", phone).one();         // 5. 判断用户是否存在         if(user == null){             // 6. 不存在,创建新用户并保存user = createUserWithPhone(phone);         }// 7. 保存用户信息到 session 中(通过 BeanUtil.copyProperties 方法将 user 中的信息过滤到 UserDTO 上,即用来隐藏部分信息)         session.setAttribute("user", BeanUtil.copyProperties(user, UserDTO.class));         return Result.ok();     }     private User createUserWithPhone(String phone) {         // 1. 创建用户         User user = new User();         user.setPhone(phone);         user.setNickName("user_" + RandomUtil.randomString(10));         // 2. 保存用户(这里使用 mybatis-plus)         save(user);         returnuser;     } }1.4 实现登录校验-

登录校验-实现:

public class LoginInterceptor implements HandlerInterceptor {     @Override     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throwsException{         // 1. 获取 session         HttpSession session = request.getSession();         // 2. 获取 session 中的用户UserDTO user = (UserDTO) session.getAttribute("user");         // 3. 判断用户是否存在         if(user == null){             // 4. 不存在,拦截,返回 401 未授权response.setStatus(401);             return false;         }         // 5. 存在,保存用户信息到 ThreadLocalUserHolder.saveUser(user);// 6. 放行         return true;     }     @Override     public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {         // 移除用户,避免内存泄露UserHolder.removeUser();     } }

UserHolder 类的实现: 该类中定义了一个静态的 ThreadLocal

public class UserHolder {     private static final ThreadLocal<UserDTO> tl = new ThreadLocal<>();     public static void saveUser(UserDTO user){         tl.set(user);     }     public static UserDTO getUser(){         return tl.get();     }public static void removeUser(){         tl.remove();     } }

配置-:

@Configuration public class MvcConfig implements WebMvcConfigurer {     @Override     public void addInterceptors(InterceptorRegistry registry) {         registry.addInterceptor(newLoginInterceptor())                 .excludePathPatterns("/user/login",                         "/user/code"                 );     } }

前端请求说明:

说明请求方式POST请求路径/user/me请求参数无返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Override     public Result me() {         UserDTO user = UserHolder.getUser();         return Result.ok(user);     } }

2. 集群的 session 共享问题

session 共享问题:

多台 tomcat 并不共享 session 存储空间,当请求切换到不同 tomcat 服务时会导致数据丢失的问题。

session 的替代方案应该满足以下条件:

数据共享(不同的 tomcat 都能够访问 Redis 中的数据)

内存存储(Redis 通过内存存储)

key、value 结构(Redis 是 key-value 结构)

3. 基于 Redis 实现共享 session 登录

3.1 Redis 实现共享 session 登录流程图3.2 实现发送短信验证码

前端请求说明:

说明请求方式POST请求路径/user/code请求参数phone(电话号码)返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Resource     privateStringRedisTemplate stringRedisTemplate;@Override     public Result sendCode(String phone, HttpSession session) {         // 1. 校验手机号         if(RegexUtils.isPhoneInvalid(phone)) {// 2. 如果不符合,返回错误信息             return Result.fail("手机号格式错误!");         }// 3. 符合,生成验证码(设置生成6位)         String code = RandomUtil.randomNumbers(6);         // 4. 保存验证码到 Redis(以手机号为 key,设置有效期为 2min)         stringRedisTemplate.opsForValue().set("login:code:" + phone, code, 2, TimeUnit.MINUTES);         // 5. 发送验证码(这里并未实现,通过日志记录)         log.debug("发送短信验证码成功,验证码:{}", code);         // 返回 ok         returnResult.ok();     } }3.3 实现短信验证码登录、注册

前端请求说明:

说明请求方式POST请求路径/user/login请求参数phone(电话号码);code(验证码)返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Override     public Result login(LoginFormDTO loginForm, HttpSession session) {         // 1. 校验手机号String phone = loginForm.getPhone();if(RegexUtils.isPhoneInvalid(phone)){             // 不一致,返回错误信息             return Result.fail("手机号格式错误!");         }// 2. 校验验证码         String cacheCode = (String) session.getAttribute("code");         String code = loginForm.getCode();if(cacheCode == null || !cacheCode.equals(cacheCode)){             // 不一致,返回错误信息             returnResult.fail("验证码错误!");         }         // 4. 一致,根据手机号查询用户(这里使用的 mybatis-plus)         User user = query().eq("phone", phone).one();         // 5. 判断用户是否存在         if(user == null){             // 6. 不存在,创建新用户并保存user = createUserWithPhone(phone);         }// 7. 保存用户信息到 session 中(通过 BeanUtil.copyProperties 方法将 user 中的信息过滤到 UserDTO 上,即用来隐藏部分信息)         session.setAttribute("user", BeanUtil.copyProperties(user, UserDTO.class));         returnResult.ok();     }private User createUserWithPhone(String phone) {         // 1. 创建用户         User user = newUser();         user.setPhone(phone);         user.setNickName("user_" + RandomUtil.randomString(10));         // 2. 保存用户(这里使用 mybatis-plus)         save(user);         return user;     } }3.4 实现登录校验-

这里将原有的一个-分成两个-,第一个-对所有的请求进行拦截,每次拦截刷新 token 的有效期,并将能查询到的用户信息保存到 ThreadLocal 中。第二个-则进行拦截功能,对需要登录的路径进行拦截。

刷新 token -实现:

public class RefreshTokenInterceptor implements HandlerInterceptor {     privateStringRedisTemplate stringRedisTemplate;public RefreshTokenInterceptor(StringRedisTemplate stringRedisTemplate){         this.stringRedisTemplate = stringRedisTemplate;     }@Override     public booleanpreHandle(HttpServletRequest request, HttpServletResponse response,Object handler) throws Exception {         // 1. 获取请求头中的 token         Stringtoken = request.getHeader("authorization");         if (StrUtil.isBlank(token)) {             return true;         }         // 2. 基于 token 获取 redis 中的用户         String tokenKey = "login:token:" + token;         Map<ObjectObject> userMap = stringRedisTemplate.opsForHash().entries(tokenKey);// 3. 判断用户是否存在         if (userMap.isEmpty()) {             return true;         }         // 5. 将查询到的 Hash 数据转为 UserDTO 对象UserDTO user = BeanUtil.fillBeanWithMap(userMap,new UserDTO(), false);         // 6. 存在,保存用户信息到 ThreadLocal         UserHolder.saveUser(user);         // 7. 刷新 token 有效期 30 min         stringRedisTemplate.expire(tokenKey, 30, TimeUnit.MINUTES);         // 8. 放行         return true;     } }

登录校验-实现:

public class LoginInterceptor implements HandlerInterceptor {     @Override     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {         // 1. 获取 session         HttpSession session = request.getSession();         // 2. 获取 session 中的用户UserDTO user = (UserDTO) session.getAttribute("user");         // 3. 判断用户是否存在         if(user == null){             // 4. 不存在,拦截,返回 401 未授权response.setStatus(401);             return false;         }         // 5. 存在,保存用户信息到 ThreadLocal         UserHolder.saveUser(user);         // 6. 放行         return true;     }     @Override     public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {         // 移除用户,避免内存泄露UserHolder.removeUser();     } }

UserHolder 类的实现: 该类中定义了一个静态的 ThreadLocal

public class UserHolder {     private staticfinal ThreadLocal<UserDTO> tl =new ThreadLocal<>();     public static void saveUser(UserDTO user){         tl.set(user);     }     public static UserDTO getUser(){         return tl.get();     }     public static void removeUser(){         tl.remove();     } }

配置-:

@Configuration public class MvcConfig implements WebMvcConfigurer {     @Resource     private StringRedisTemplate stringRedisTemplate;     @Override     public void addInterceptors(InterceptorRegistry registry) {         registry.addInterceptor(new RefreshTokenInterceptor(stringRedisTemplate))                 .addPathPatterns("/**").order(0);         registry.addInterceptor(newLoginInterceptor())                 .excludePathPatterns("/user/login",                         "/user/code"                 ).order(1);     } }

前端请求说明:

说明请求方式POST请求路径/user/me请求参数无返回值无

后端接口实现:

@Slf4j @Service public class UserServiceImpl extends ServiceImpl<UserMapperUserimplements IUserService {     @Override     public Result me() {         UserDTO user = UserHolder.getUser();         return Result.ok(user);     } }

“Redis的共享session应用如何实现短信登录”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注网站,小编将为大家输出更多高质量的实用文章!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:MySQL中流式查询及游标查询的方式是什么
下一篇:MySQL存储过程的参数如何使用
相关文章

 发表评论

暂时没有评论,来抢沙发吧~