0
点赞
收藏
分享

微信扫一扫

【项目】小帽外卖(六)

Separes 2022-04-21 阅读 57
java

小帽外卖

第六章 手机验证码登录

一、效果展示

在这里插入图片描述

二、短信发送

1. 短信服务介绍

  • 目前市面上有很多第三方提供的短信服务,这些第三方短信服务会和各个运营商(移动、联通、电信)对接,我们只需要注册成为会员并且按照提供的开发文档进行调用就可以发送短信。需要说明的是,这些短信服务一般都是收费服务。
  • 常用短信服务:
    • 阿里云,华为云,腾讯云,京东,梦网,乐信

2. 阿里云短信服务 - 介绍

  • 阿里云短信服务(Short Message Service) 是广大企业客户快速触达手机用户所优选使用的通信能力。调用API或用群发助手,即可发送验证码、通知类和营销类短信;国内验证短信秒级触达,到达率最高可达99%;国际/港澳台短信覆盖200多个国家和地区,安全稳定,广受出海企业选用。
  • 应用场景:
    • 验证码
    • 短信通知
    • 推广短信
      在这里插入图片描述

3. 阿里云短信服务 - 注册账号

在这里插入图片描述

4. 阿里云短信服务 - 设置短信签名

  • 注册成功后,点击登录按钮进行登录。登录后进入短信服务管理页面,选择国内信息菜单:
    在这里插入图片描述
  • 短信签名是短信发送者的署名,表示发送方的身份。

5. 阿里云短信服务 - 设置短信模板

  • 切换到【模板管理】标签页:
    在这里插入图片描述
  • 短信模板包含短信发送内容、场景、变量信息。

6. 阿里云短信服务 - 设置AccessKey

  • 光标移动到用户头像上,在弹出的窗口中点击【AccessKey管理】:
    在这里插入图片描述

7. 代码开发

  • 使用阿里云信息服务发送短信,可以参照官方提供的文档即可。
  • 具体开发步骤:
    • 导入maven坐标
    • 调用API
7.1 导入maven坐标

在这里插入图片描述

7.2 调用API

在这里插入图片描述

三、手机验证码登录

1. 需求分析

  • 为了方便用户登录,移动端通常都会提供通过手机验证码登录的功能。
  • 手机验证码登录的优点:
    • 方便快捷,无需注册,直接登录
    • 使用短信验证码作为登录凭证,无需记忆密码
    • 安全
  • 登录流程:
    • 输入手机号>获取验证码>输入验证码>点击登录>登录成功
  • 注意:通过手机验证码登录,手机号是区分不同用户的标识。
    在这里插入图片描述

2. 数据模型

  • 通过手机验证码登录时,涉及的表为user表,即用户表。结构如下:
    在这里插入图片描述

3. 代码开发

3.1 梳理交互过程
  • 在开发代码之前,需要梳理一下登录时前端页面和服务端的交互过程:
    • 在登录页面(front/page/login.html)输入手机号,点击【获取验证码】按钮,页面发送ajax请求,在服务端调用短信服务API给指定手机号发送验证码短信
    • 在登录页面输入验证码,点击【登录】按钮,发送ajax请求,在服务端处理登录请求
  • 开发手机验证码登录功能,其实就是在服务端编写代码去处理前端页面发送的这2次请求即可。
3.2 准备工作
  • 在开发业务功能前,先将需要用到的类和接口基本结构创建好:
    • 实体类 User
    • Mapper接口 UserMapper
    • 业务层接口 UserService
    • 业务层实现类 UserServiceImpl
    • 控制层 UserController
    • 工具类 SMSUtils、ValidateCodeUtils
3.3 修改LoginCheckFilter
  • 前面我们已经完成了LoginCheckFilter过滤器的开发,此过滤器用于检查用户的登录状态。我们在进行手机验证码登录时,发送的请求需要在此过滤器处理时直接放行。
    在这里插入图片描述
  • 在LoginCheckFilter过滤器中扩展逻辑,判断移动端用户登录状态:
    在这里插入图片描述
@WebFilter(filterName = "loginCheckFilter", urlPatterns = "/*")
@Slf4j
public class LoginCheckFilter implements Filter {
    // 路径匹配器,支持通配符
    public static final AntPathMatcher PATH_MATCHER = new AntPathMatcher();
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        // 1.获取本次请求的URI
        String requestURI = request.getRequestURI();    // /backend/index.html

        log.info("拦截到请求:{}", requestURI);

        // 定义不需要处理的请求路径
        String[] urls = new String[] {
                "/employee/login",
                "/employee/logout",
                "/backend/**",
                "/front/**",
                "/common/**",
                "/user/sendMsg",    // 移动端发送短信
                "/user/login"       // 移动端登录
        };
        // 2.判断本次请求是否需要处理
        boolean check = check(urls, requestURI);
        // 3.如果不需要处理,则直接放行
        if(check) {
            log.info("本次请求{}不需要处理", requestURI);
            filterChain.doFilter(request, response);
            return;
        }
        // 4-1.判断登录状态,如果已登录,则直接放行
        if(request.getSession().getAttribute("employee") != null) {
            log.info("用户已登录,用户id为:{}", request.getSession().getAttribute("employee"));
            Long empId = (Long) request.getSession().getAttribute("employee");
            BaseContext.setCurrentId(empId);
            filterChain.doFilter(request, response);
            return;
        }

        // 4-2.判断登录状态,如果已登录,则直接放行
        if(request.getSession().getAttribute("user") != null) {
            log.info("用户已登录,用户id为:{}", request.getSession().getAttribute("user"));
            Long userId = (Long) request.getSession().getAttribute("user");
            BaseContext.setCurrentId(userId);
            filterChain.doFilter(request, response);
            return;
        }

        log.info("用户未登录");
        // 5.如果未登录则返回未登录结果,通过输出流方式向客户端页面响应数据
        response.getWriter().write(JSON.toJSONString(R.error("NOTLOGIN")));
        return;
    }

    /**
     * 路径匹配,检查本次请求是否需要放行
     * @param urls
     * @param requestURI
     * @return
     */
    public boolean check(String[] urls, String requestURI) {
        for(String url : urls) {
            boolean match = PATH_MATCHER.match(url, requestURI);
            if(match) {
                return true;
            }
        }
        return false;
    }
}
@RestController
@RequestMapping("/user")
@Slf4j
public class UserController {
    @Autowired
    private UserService userService;

    /**
     * 发送手机短信验证码
     * @param user
     * @return
     */
    @PostMapping("/sendMsg")
    public R<String> sendMsg(@RequestBody User user, HttpSession session) {
        // 获取手机号
        String phone = user.getPhone();

        if(StringUtils.isNotEmpty(phone)) {
            // 生成随机的4位验证码
            String code = ValidateCodeUtils.generateValidateCode(4).toString();
            log.info("code={}",code);
            // 调用阿里云提供的短信服务API完成发送短信
            // SMSUtils.sendMessage("小帽外卖", "", phone, code);
            // 需要将生成的验证码保存到Session
            session.setAttribute(phone, code);
            return R.success("手机验证码短信发送成功");
        }

        return R.error("短信发送失败");
    }

    /**
     * 移动端用户登录
     * @param map
     * @param session
     * @return
     */
    @PostMapping("/login")
    public R<User> sendMsg(@RequestBody Map map, HttpSession session) {
        log.info(map.toString());
        // 获取手机号
        String phone = map.get("phone").toString();
        // 获取验证码
        String code = map.get("code").toString();
        // 从Session中获取保存的验证码
        Object codeInSession = session.getAttribute(phone);
        // 进行验证码的比对(页面提交的验证码和Session中保存的验证码比对)
        if(codeInSession != null && codeInSession.equals(code)) {
            // 如果能够比对成功,说明登录成功
            // 判断当前手机号对应的用户是否为新用户,如果是新用户就自动完成注册
            LambdaQueryWrapper<User> queryWrapper = new LambdaQueryWrapper<>();
            queryWrapper.eq(User::getPhone, phone);
            User user = userService.getOne(queryWrapper);
            if(user == null) {
                // 判断当前手机号对应的用户是否为新用户,如果是新用户就自动完成注册
                user = new User();
                user.setPhone(phone);
                user.setStatus(1);
                userService.save(user);
            }
            session.setAttribute("user", user.getId());
            return R.success(user);
        }

        return R.error("登录失败");
    }
}
举报

相关推荐

0 条评论