目录

  • 前言
    • UserAction.java
    • JwtFilter
    • Login.vue
  • 后记

前言

本次只是把验证码的验证搞了一下,

思路:

  1. 从后端随机生成验证码,存储到redis里面,同时发送到前台(在进入登录界面的时候)

  2. 前台把验证码展示出来,在提交表单时,把用户输入的验证码作为参数传输。

  3. 后端接收到前台传过来的验证码,同时从redis拿值相对比,就能判断是否正确。

UserAction.java

我多写了一个生成随机数的方法,同时我修改了登录的条件。

package com.liwangwang.vue.web;import java.util.HashMap;
import java.util.Map;
import java.util.Random;import com.fasterxml.jackson.databind.ObjectMapper;
import com.liwangwang.base.web.BaseAction;
import com.liwangwang.vue.biz.UserBiz;
import com.liwangwang.vue.entity.User;
import com.liwangwang.vue.util.JsonData;
import com.liwangwang.vue.util.JwtUtils;
import com.liwangwang.vue.util.ResponseUtil;
import com.liwangwang.vue.util.StringUtils;
import com.opensymphony.xwork2.ModelDriven;import redis.clients.jedis.Jedis;public class UserAction extends BaseAction implements ModelDriven<User>{private UserBiz userBiz;private User user = new User();public UserBiz getUserBiz() {return userBiz;}public void setUserBiz(UserBiz userBiz) {this.userBiz = userBiz;}private static Jedis jedis;static {jedis = new Jedis("192.168.219.129", 6379);jedis.auth("123456");//权限认证jedis.select(1);}public String login() {ObjectMapper om = new ObjectMapper();JsonData jsonData = null;try {String userCode = request.getParameter("userCode");//从Redis里面拿值String code = jedis.get("code");System.out.println(userCode);System.out.println(code);if(StringUtils.isBlank(user.getUname()) || StringUtils.isBlank(user.getPwd())) {jsonData = new JsonData(0, "用户或者密码为空", user);}else if(!code.equals(userCode) || userCode==null ) {jsonData = new JsonData(0, "验证码错误或为空", user);}else {User u = this.userBiz.login(user);Map<String, Object> claims = new HashMap<String, Object>();claims.put("uname",user.getUname());claims.put("pwd", user.getPwd());String jwt = JwtUtils.createJwt(claims, JwtUtils.JWT_WEB_TTL);response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);jsonData = new JsonData(1, "登录成功", u);}} catch (Exception e) {e.printStackTrace();jsonData = new JsonData(0, "用户或者密码错误", user);}finally {try {ResponseUtil.write(response, om.writeValueAsString(jsonData));} catch (Exception e) {e.printStackTrace();}}return null;}/*** 生成随机数* @return*/public String userCode() {Random ra = new Random();String sCode = "A,B,C,E,F,G,H,J,K,L,M,N,P,Q,R,S,T,W,X,Y,Z,1,2,3,4,5,6,7,8,9,0";String[] split = sCode.split(",");String sb = new String();for (int i = 0; i < 4; i++) {System.out.println(ra.nextInt(30));sb+=(split[ra.nextInt(30)]);}ObjectMapper om = new ObjectMapper();JsonData jsonData = null;try {jsonData = new JsonData(0, sb, user);ResponseUtil.write(response, om.writeValueAsString(jsonData));} catch (Exception e) {System.out.println(e);}//把验证码放到Redis缓存里jedis.set("code", sb);return null;}public static void main(String[] args) {UserAction us = new UserAction();us.userCode();}public String getAsyncData() {ObjectMapper om = new ObjectMapper();try {Thread.sleep(6000);ResponseUtil.write(response, om.writeValueAsString("http://www.javaxl.com"));} catch (Exception e) {e.printStackTrace();}return null;}@Overridepublic User getModel() {return user;}
}

JwtFilter

没多干什么,就是多添加了一个验证码的通道。

有了这个通道,就可以无视登录也可以访问后台资源。

package com.liwangwang.vue.util;import java.io.IOException;
import java.util.regex.Matcher;
import java.util.regex.Pattern;import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import io.jsonwebtoken.Claims;/*** * JWT验证过滤器,配置顺序 :CorsFilte-->JwtFilter-->struts2中央控制器* * @author Administrator**/
public class JwtFilter implements Filter {// 排除的URL,一般为登陆的URL(请改成自己登陆的URL)private static String EXCLUDE = "^/vue/userAction_login\\.action?.*$";private static Pattern PATTERN = Pattern.compile(EXCLUDE);private boolean OFF = false;// true关闭jwt令牌验证功能@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void destroy() {}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {HttpServletRequest req = (HttpServletRequest) request;HttpServletResponse resp = (HttpServletResponse) response;String path = req.getServletPath();if (OFF || isExcludeUrl(path)) {// 登陆直接放行chain.doFilter(request, response);return;}//新增的一个通道,用来放验证码的if(OFF || Pattern.compile("^/vue/userAction_userCode\\.action?.*$").matcher(path).matches()) {chain.doFilter(request, response);return;}// 从客户端请求头中获得令牌并验证String jwt = req.getHeader(JwtUtils.JWT_HEADER_KEY);Claims claims = this.validateJwtToken(jwt);if (null == claims) {// resp.setCharacterEncoding("UTF-8");resp.sendError(403, "JWT令牌已过期或已失效");return;} else {String newJwt = JwtUtils.copyJwt(jwt, JwtUtils.JWT_WEB_TTL);resp.setHeader(JwtUtils.JWT_HEADER_KEY, newJwt);chain.doFilter(request, response);}}/*** 验证jwt令牌,验证通过返回声明(包括公有和私有),返回null则表示验证失败*/private Claims validateJwtToken(String jwt) {Claims claims = null;try {if (null != jwt) {claims = JwtUtils.parseJwt(jwt);}} catch (Exception e) {e.printStackTrace();}return claims;}/*** 是否为排除的URL* * @param path* @return*/private boolean isExcludeUrl(String path) {Matcher matcher = PATTERN.matcher(path);return matcher.matches();}// public static void main(String[] args) {// String path = "/sys/userAction_doLogin.action?username=zs&password=123";// Matcher matcher = PATTERN.matcher(path);// boolean b = matcher.matches();// System.out.println(b);// }}

Login.vue

/*** 对后台请求的地址的封装,URL格式如下:* 模块名_实体名_操作*/
export default {'SERVER': 'http://localhost:8080/sshdemo', //服务器'SYSTEM_USER_DOLOGIN': '/vue/userAction_login.action', //用户登陆'SYSTEM_USER_USERCODE': '/vue/userAction_userCode.action', //验证码验证'SYSTEM_MENU_TREE_LIST': '/vue/treeNodeAction_selectbylist.action', //树形的另一个'SYSTEM_USER_DOREG': '/vue/userAction_reg.action', //用户注册'SYSTEM_MENU_TREE': '/vue/treeNodeAction_selectlist.action', //左侧树形菜单加载'SYSTEM_ARTICLE_LIST': '/vue/articleAction_list.action', //文章列表'SYSTEM_ARTICLE_ADD': '/vue/articleAction_add.action', //文章新增'SYSTEM_ARTICLE_EDIT': '/vue/articleAction_edit.action', //文章修改'SYSTEM_ARTICLE_DEL': '/vue/articleAction_del.action', //文章删除'SYSTEM_USER_GETASYNCDATA': '/vue/userAction_getAsyncData.action', //vuex中的异步加载数据'getFullPath': k => { //获得请求的完整地址,用于mockjs测试时使用return this.SERVER + this[k];}
}

Login.vue

<template><div class="login-wrap"><el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container"><h3 style="text-align: center;">用户登录</h3><el-form-item label="用户名"><el-input type="text" v-model="ruleForm.uname"></el-input></el-form-item><el-form-item label="密码"><el-input type="password" v-model="ruleForm.pwd"></el-input></el-form-item><el-form-item label="验证码"><el-input type="text" v-model="ruleForm.userCode"></el-input>{{code}}</el-form-item><el-form-item><el-row><el-col :span="24"><div class="grid-content bg-purple-dark"><el-button type="primary" style="width:100%;" @click="doLogin">提交</el-button></div></el-col></el-row><el-row><el-col :span="12"><div class="grid-content bg-purple-dark"><el-link type="success" @click="toReg">用户注册</el-link></div></el-col><el-col :span="12"><div class="grid-content bg-purple-dark"><el-link type="success">忘记密码</el-link></div></el-col></el-row></el-form-item></el-form></div>
</template><script>export default {data() {return {ruleForm: {uname: '',pwd: '',userCode:''},code:null};},created(){let url = this.axios.urls.SYSTEM_USER_USERCODE;this.axios.post(url,{}).then((response)=>{console.log(response);this.code = response.data.msg;}).catch(function(error){console.log(error);});},methods: {toReg() {this.$router.push({path: '/Reg'});},doLogin() {let url = this.axios.urls.SYSTEM_USER_DOLOGIN;console.log(url);this.axios.post(url, this.ruleForm).then((response)=> {console.log(response.data);if (response.data.code == 1) {this.$message({showClose: true,message: response.data.msg,type: 'success'});this.$router.push({path:'/AppMain'})} else {this.$message({showClose: true,message: response.data.msg,type: 'error'});}}).catch(function(error) {console.log(error);});}}}
</script><style scoped>.login-wrap {box-sizing: border-box;width: 100%;height: 100%;padding-top: 10%;background-image: url();/* background-color: #112346; */background-repeat: no-repeat;background-position: center right;background-size: 100%;}.login-container {border-radius: 10px;margin: 0px auto;width: 350px;padding: 30px 35px 15px 35px;background: #fff;border: 1px solid #eaeaea;text-align: left;box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);}.title {margin: 0px auto 40px auto;text-align: center;color: #505458;}
</style>

后记

用redis存储的验证码相关推荐

  1. ssh,ssm整合redis存储验证码

    ssh整合redis存储验证码 SSM整合redis存储验证码 思路 **获取验证码部分:** **登录部分** 细节 详细代码 前端 后端 SSH 思路: 测试redis是否连接成功 详细代码 后端 ...

  2. 搭建企业级微信公众号管理平台(三)----注册码实现与校验,Redis存储

    1.统一接口返回结果集格式 接口应用 /*** 功能说明: 获取微信服务接口*/@ApiOperation(value = "微信应用服务接口")@GetMapping(" ...

  3. Redis——模拟手机验证码发送

    使用redis模拟手机验证码发送,规定一个手机号一天只能请求三次验证码,且每次请求的验证码只有两分钟就会过期 1.首先写一个函数随机产生六位验证码,使用random函数 //随机生成六位验证码publ ...

  4. 每秒1w+分布式事务--dtm的Redis存储性能测试分析

    概述 之前dtm给出了Mysql作为存储引擎的性能测试报告,在一个普通配置的机器上,2.68w IOPS,4核8G机器上,能够支持大约每秒900+分布式事务,能够满足大部分公司的业务需求. 此次带来的 ...

  5. Redis存储总是心里没底?你大概漏了这些数据结构原理

    上一篇文章<Redis存储总用String?你大概错过了更优的使用方法>我们了解了Redis的数据类型特点与适用场景,本期内容则会着重讲讲Redis内存数据结构与编码,弄清Redis内部到 ...

  6. 阿里云短信验证解决方案(java版)(redis存储)

    阿里云短信验证解决方案(java版)(redis存储) 参考文章: (1)阿里云短信验证解决方案(java版)(redis存储) (2)https://www.cnblogs.com/Amos-Tur ...

  7. 大容量类 Redis 存储的场景补充-pika

    2019独角兽企业重金招聘Python工程师标准>>> 导读 我们在<大容量类 Redis 存储 - 有关 pika 的一切>里介绍过pika的诞生.pika的特点.pi ...

  8. Redis 存储字符串和对象

    今天用redis存储,发现客户端jedis提供的存储方法中存储的类型只有String和byte数据,没有能够存储对象的,网上发现可以序列化存储对象.这就开始了我第一次序列化之旅. 1 测试类 impo ...

  9. tomcat使用redis存储共享session

    tomcat使用redis存储共享session 在tomcat集群环境下实现session共享有几种解决方式,这里介绍一种简单的方案. 使用redis对session进行存储,配置比較简单.webs ...

最新文章

  1. vfp中,函数subs(计算机管理信息系统,7)返回的结果是,VFP选择题库(可发学生).xls...
  2. 腾讯年度性爱报告发布,最后一条数据羞愧了...
  3. Servlet的第一个程序HelloWorld
  4. 牛客 - Across the Firewall(最大流)
  5. 【Servlet】Filter过滤器详解、使用示例
  6. php warning date(),lnmp打开cacti时提示PHP Warning: date()
  7. 阮一峰react demo代码研究的学习笔记 - demo5 debug
  8. 【渝粤题库】陕西师范大学200991教育统计学作业(高起本)
  9. JavaScript————FormData实现多文件上传
  10. 【华为云技术分享】【DevCloud•敏捷智库】读懂敏捷需求管理的4个关键词
  11. 关于G++库链接的一个问题
  12. 成功解决 AttributeError: module ‘neat’ has no attribute ‘Config’解决方式
  13. IDEA系列(六)一This file is indented with tabs instead of 4 space
  14. js基础知识汇总08
  15. k近邻算法_机器学习算法之——K最近邻(k-Nearest Neighbor,KNN)分类算法原理讲解...
  16. matlab textscan读结构化.txt文件
  17. 扫盲区分IaaS和PaaS
  18. nim语言编译后的c语言,Nim的编译方法
  19. 闲鱼搜索召回升级:向量召回个性化召回
  20. 网站开发之HTML基础知识及超链接(二)

热门文章

  1. 如何用html语言浏览 路径,小白编程系列(二):前端学习路径
  2. jQuery方法报错...is not a function问题
  3. 转:领导力的核心是让每个人得以蓬勃发展
  4. OPPO R15 Pro在哪里打开USB调试模式的简单经验
  5. Fiddler_IE浏览器打开网页提示安全证书过期或证书错误的解决方法
  6. 英特尔® 嵌入式显卡驱动程序
  7. html制作古诗带图画大全,古诗词的手抄报图画设计模板
  8. 2022最全软件测试Linux面试题(含答案)
  9. 抖音风格故障效果PSD图层+剪辑设计必备
  10. iOS四种方法读取文件内容