> 文章列表 > 验证码登录开发----手机验证码登录

验证码登录开发----手机验证码登录

验证码登录开发----手机验证码登录

手机验证码登录

需求分析

为了方便用户登录,移动端通常都会提供通过手机验证码登录的功能

手机验证码登录的优点:

  • 方便快捷、无需注册,直接登录
  • 使用短信验证码作为登录凭证,无需记忆密码
  • 安全

登录流程:

输入手机号>获取验证码>输入验证码>点击登录>登录成功

注意:通过手机验证码登录,手机号是区分不同用户的标识

代码开发–梳理交互过程

在开发代码之前,需要梳理一下登录时前端页面和服务端的交互过程:

  1. 在登录页面(front/page/login.html)输入手机号,点击【获取验证码】按钮,页面发送ajax请求,在服务端调用短信服务API给指定手机号发送验证码短信
  2. 在登录页面输入验证码,点击【登录】按钮,发送ajax请求,在服务端处理登录请求

开发手机验证码登录功能,其实就是在服务端编写代码去处理前端页面发送的这2次请求即可

代码开发–准备工作

在开发业务功能前,先将需要用到的类和接口基本结构创建好

  • 实体类User

    /*** 用户信息*/
    @Data
    public class User implements Serializable {private static final long serialVersionUID = 1L;private Long id;//姓名private String name;//手机号private String phone;//性别 0 女 1 男private String sex;//身份证号private String idNumber;//头像private String avatar;//状态 0:禁用,1:正常private Integer status;
    }
  • Mapper接口UserMapper

  • 业务层接口UserService

  • 业务层实现类UserServiceImpl

  • 控制层UserController

    @Slf4j
    @RestController
    @RequestMapping("/user")
    public class UserController {@Autowiredprivate UserService userService;/** 发送手机短信验证码* */@PostMapping("/sendMsg")public R<String> sendMsg(@RequestBody User user, HttpSession session){//获取手机号String phone = user.getPhone();if(StringUtils.isNotEmpty(phone)){//生成随机的4位验证码String validateCode = ValidateCodeUtils.generateValidateCode(4).toString();log.info("ValidateCode:{}",validateCode);//调用阿里云提供的短信服务API完成发送短信//SMSUtils.sendMessage("申请签名","模板code",phone,validateCode);//需要将生成的验证码保存到Sessionsession.setAttribute(phone,validateCode);return R.success("手机验证短信发送成功!!!");}return R.error("短信发送失败!!!");}/** 移动端用户登录* */@PostMapping("/login")public R<User> login(@RequestBody Map map, HttpSession session){/*这里会出现一个问题前端传过来的验证码User类不能接收* 有两种解决方法:* 1.我们可以创建一个UserDto,继承User* 2.还有一种就是可以使用Map来接收,用键值对的接收* */log.info(map.toString());//首先获取手机号String phone = map.get("phone").toString();//其次获取验证码String code = map.get("code").toString();//从session中获得保存到的验证码Object codeSession = session.getAttribute(phone);//进行验证码的比对(页面提交的验证码和Session中保存的验证码比对)if (codeSession!=null&&codeSession.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);//设置状态 1表示正常userService.save(user);}session.setAttribute("user",user.getId());//经过过滤器会去校验,所以我们需要将用户id存储到session中去return R.success(user);}return R.error("登录失败!!!");}
    }
    
  • 工具类SMSUtils、ValidateCodeUtils

    • SMSUtils

      /*** 短信发送工具类*/
      public class SMSUtils {/*** 发送短信* @param signName 签名* @param templateCode 模板* @param phoneNumbers 手机号* @param param 参数*/public static void sendMessage(String signName, String templateCode,String phoneNumbers,String param){DefaultProfile profile = DefaultProfile.getProfile("cn-hangzhou", "", "");IAcsClient client = new DefaultAcsClient(profile);SendSmsRequest request = new SendSmsRequest();request.setSysRegionId("cn-hangzhou");request.setPhoneNumbers(phoneNumbers);request.setSignName(signName);request.setTemplateCode(templateCode);request.setTemplateParam("{\\"code\\":\\""+param+"\\"}");try {SendSmsResponse response = client.getAcsResponse(request);System.out.println("短信发送成功");}catch (ClientException e) {e.printStackTrace();}}}
    • ValidateCodeUtils

    /*** 随机生成验证码工具类*/public class ValidateCodeUtils {/*** 随机生成验证码* @param length 长度为4位或者6位* @return*/public static Integer generateValidateCode(int length){Integer code =null;if(length == 4){code = new Random().nextInt(9999);//生成随机数,最大为9999if(code < 1000){code = code + 1000;//保证随机数为4位数字}}else if(length == 6){code = new Random().nextInt(999999);//生成随机数,最大为999999if(code < 100000){code = code + 100000;//保证随机数为6位数字}}else{throw new RuntimeException("只能生成4位或6位数字验证码");}return code;}/*** 随机生成指定长度字符串验证码* @param length 长度* @return*/public static String generateValidateCode4String(int length){Random rdm = new Random();String hash1 = Integer.toHexString(rdm.nextInt());String capstr = hash1.substring(0, length);return capstr;}}

代码开发–修改LoginCheckFilter

前面我们已经完成了LoginCheckFilter过滤器的开发,此过滤器用于检查用户的登录状态。我们在进行手机验证码登录时,发送的请求需要在此过滤器处理时直接放行

LoginCheckFilter过滤器的编写链接

@WebFilter(filterName = "LoginCheckFilter",urlPatterns = "/*")//利用过滤器拦截,拦截所有的请求  urlPatterns = "/*"
@Slf4j
public class LoginCheckFilter implements Filter {//路径匹配器,并支持通配符public static final AntPathMatcher PATH_MATCHER = new AntPathMatcher();@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest)servletRequest;HttpServletResponse response = (HttpServletResponse)servletResponse;//1. 获取本次请求的URIString requestURI = request.getRequestURI();//如果请求路径为"/backend/index.html"会与放行的"/backend/**"路径匹配不上,会导致无法识别,这时需要路径匹配器log.info("拦截到的请求:{}",requestURI);//1.2 定义不需要处理的请求路径,比如:登录页面,退出,静态资源,主要是拦截controller资源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 employeeId =(Long) request.getSession().getAttribute("employee");BaseContext.setCurrentId(employeeId);//保存用户id,用于公共字段自动填充使用long id = Thread.currentThread().getId();log.info("线程id为:{}",id);//已经登录,直接放行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);//保存用户id,用于公共字段自动填充使用long id = Thread.currentThread().getId();log.info("线程id为:{}",id);//已经登录,直接放行filterChain.doFilter(request,response);//放行return;}//5. 如果未登录则返回未登录结果,通过输出流的方式向客户端响应数据log.info("用户未登录");response.getWriter().write(JSON.toJSONString(R.error("NOTLOGIN")));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;}
}

在这里插入图片描述