当前位置: 首页 > news >正文

基于SpringBoot和Redis实现短信验证码功能

听说微信搜索《Java鱼仔》会变更强哦!

本文收录于JavaStarter ,里面有我完整的Java系列文章,学习或面试都可以看看哦

(一)概述

在日常使用各种app或者登陆网站的时候,基本都会看到短信验证码这个功能,实现短信验证码的方式有很多,这里给出基于SpringBoot和Redis的短信验证码实现方式。源码在文末

(二)思路

把一切都简化,短信验证码的实现无法就是下面几点:

1、后端随机生成短信验证码,并在服务器端保存一定时间(一般是5分钟)。

2、将短信验证码发给用户。

3、用户输入短信验证码提交后,在后端与之前生成的短信验证码作比较,如果相同说明验证成功,否则验证失败。

其中在服务器端保存一定时间的这个操作我一下子就想到了redis,那就用redis来实现吧!

如果你还不熟悉如何在SpringBoot中使用redis,可以快速翻阅一下我之前的一篇文章:
redis入门到精通系列(十):springboot集成redis及redis工具类的编写

(三)搭建项目

首先搭建一个SpringBoot项目,这一步就不讲了,如果你还有问题可以直接看我的SpringBoot系列。

3.1 引入依赖

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-data-redis</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-thymeleaf</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
</dependencies>

这里只需要引入上面三个依赖即可。

3.2 配置Controller

新建一个包叫Controller,新建类IndexController,编写一个简单的首页跳转:

@Controller
public class IndexController {
    @RequestMapping(value = {"/","/index"},method = RequestMethod.GET)
    public String index(){
        return "index";
    }
}

此时,如果访问ip:端口/或者ip:端口/index,就会跳转到index.html这个页面上去。

3.3 编写index.html

这个页面只需要包含两个按钮和一个输入框即可,页面放在resources/templates下

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org/">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script>
</head>
<body>
<button type="button">发送验证码</button>
<input type="text" id="input_code"/>
<button type="button">验证</button>
</body>
</html>

到这里一个简单的项目前期搭建就完成了,访问一下:
在这里插入图片描述

(四)核心功能实现

4.1 配置redis

redis在正式使用前需要编写一个配置类,我在redis系列的博客中都有写,这里就直接用了,新建一个包叫做config,新建类RedisConfig,这个类主要是让redis在传输过程中不会乱码。

@Configuration
public class RedisConfig {
    //自定义的redistemplate
    @Bean(name = "redisTemplate")
    public RedisTemplate<String,Object> redisTemplate(RedisConnectionFactory factory){
        //创建一个RedisTemplate对象,为了方便返回key为string,value为Object
        RedisTemplate<String,Object> template = new RedisTemplate<>();
        template.setConnectionFactory(factory);
        //设置json序列化配置
        Jackson2JsonRedisSerializer jackson2JsonRedisSerializer=new
                Jackson2JsonRedisSerializer(Object.class);
        ObjectMapper objectMapper=new ObjectMapper();
        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        objectMapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance);
        //string的序列化
        StringRedisSerializer stringRedisSerializer=new StringRedisSerializer();
        //key采用string的序列化方式
        template.setKeySerializer(stringRedisSerializer);
        //value采用jackson的序列化方式
        template.setValueSerializer(jackson2JsonRedisSerializer);
        //hashkey采用string的序列化方式
        template.setHashKeySerializer(stringRedisSerializer);
        //hashvalue采用jackson的序列化方式
        template.setHashValueSerializer(jackson2JsonRedisSerializer);
        template.afterPropertiesSet();
        return template;
    }
}

4.2 发送验证码功能的实现

发送验证码的思路就是生成验证码,存储验证码,通过短信接口发送验证码。这里的短信接口市面上有很多,使用起来无非就是一个get或者post请求,我这里就用控制台输出代替了。

在IndexController增加一个接口,用于发送短信验证码:

@Controller
public class IndexController {

    @Autowired
    private RedisTemplate<String,Object> redisTemplate;

    @RequestMapping(value = {"/","/index"},method = RequestMethod.GET)
    public String index(){
        return "index";
    }

    @ResponseBody
    @RequestMapping(value = "/sendmessage",method = RequestMethod.GET)
    public String sendmessage(){
        //这里的userId在真实业务中通过个人身份的令牌获取,这里直接仿造一个
        String userId="1";
        //生成六位数随机验证码
        String code=getCode();
        //设置redis的key,这里设置为项目名:使用的字段:用户Id
        String redisKey="VERIFATIONCODE:CODE:"+userId;
        //将这个验证码存入redis中,并设置失效时间为5分钟
        redisTemplate.opsForValue().set(redisKey,code,300, TimeUnit.SECONDS);
        //发送短信
        boolean isSuccess=send(code);
        if (isSuccess){
            return "success";
        }else {
            return "fail";
        }
    }

    private boolean send(String code) {
        String msg="验证码为:"+code+",验证码有效期5分钟,请及时验证";
        System.out.println(msg);
        return true;
    }
    
    //生成六位随机验证码
    private static String getCode() {
        Random random = new Random();
        String result="";
        for (int i=0;i<6;i++)
        {
            result+=random.nextInt(10);
        }
        return result;
    }
}

代码注释都写了,需要注意的是redis的key值命名方式,将项目名:使用的字段:用户Id设置为key,当然也可以采用其他办法。

接着在前端的页面上给发送验证码的按钮增加一个点击事件,并通过ajax来调用接口。

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org/">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script>
</head>
<body>
<button type="button" th:onclick="sendmessage()">发送验证码</button>
<input type="text" id="input_code"/>
<button type="button">验证</button>
<script>
    sendmessage=()=>{
        var resultjson;
        $.ajax(
            {
                url: "http://localhost:8080/sendmessage",
                type: "GET",
                async:false,
                dataType:'text',
                success: function (result) {
                    resultjson=result;
                },
                error: function ( status) {
                    console.log("error")
                }
        });
        if (resultjson=='success'){
            alert("短信发送成功");
        }else{
            alert("短信发送失败,请联系管理员处理")
        }
    }
</script>
</body>
</html>

4.4 验证短信验证码

在IndexController中再增加一个接口用来验证,逻辑就是把redis中存在的值和输入值进行比较

@ResponseBody
@RequestMapping(value = "/checkCode",method = RequestMethod.GET)
public String checkCode(@RequestParam("inputCode") String inputCode){
    String userId="1";
    //设置redis的key,这里设置为项目名:使用的字段:用户Id
    String redisKey="VERIFATIONCODE:CODE:"+userId;
    String realCode= (String) redisTemplate.opsForValue().get(redisKey);
    if (realCode!=null&&realCode.equals(inputCode)){
        return "success";
    }else {
        return "fail";
    }
}

还是一样给验证按钮也加一个点击事件,checkCode方法加到sendmessage方法后。

<button type="button" th:onclick="checkCode()">验证</button>

checkCode=()=>{
    var resultjson;
    var input_code=$("#input_code").val();
    $.ajax({
            url: "http://localhost:8080/checkCode",
            type: "GET",
            async:false,
            dataType:'text',
            data:{inputCode:input_code},
            success: function (result) {
                resultjson=result;
            },
            error: function ( status) {
                console.log("error")
            }
    });
    if (resultjson=='success'){
        alert("验证成功");
    }else{
        alert("验证失败,验证码错误")
    }
}

(五)效果展现

首先点击发送验证码,页面上弹出短信发送成功,后台接收到验证码:

在这里插入图片描述

观察redis中的数据

在这里插入图片描述

首先输入一个错误的数据

在这里插入图片描述

再输入一个正确的数据
在这里插入图片描述

最后给出源码:项目源码

相关文章:

  • iOS设计模式 - 外观
  • 大厂面试题:你知道JUC中的Semaphore、CyclicBarrier、CountDownLatch吗
  • 勤能补挫-简单But易错的JSCSS问题总结
  • Java16都快上线了,你该不会连Java8的特性都不会用吧?
  • ubuntu 更改密码
  • 从考研失败到最具成长力员工,这个2020就像过山车一样
  • 解析MySQL权限表
  • 图解Mysql索引的数据结构!看不懂你来找我
  • 如何用Java写一个规范的http接口?
  • Getting started with Java EE 8 MVC(1)
  • 产品经理问我:手动创建线程不香吗,为什么非要用线程池呢?
  • 将桌面上的硬盘移除
  • 白话Mysql的锁和事务隔离级别!死锁、间隙锁你都知道吗?
  • Jquery datatables 使用方法
  • 基于SpringBoot实现文件的上传下载
  • 【Leetcode】101. 对称二叉树
  • CentOS 7 防火墙操作
  • emacs初体验
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • jquery ajax学习笔记
  • JS字符串转数字方法总结
  • Linux快速复制或删除大量小文件
  • Logstash 参考指南(目录)
  • Netty 4.1 源代码学习:线程模型
  • overflow: hidden IE7无效
  • PaddlePaddle-GitHub的正确打开姿势
  • python3 使用 asyncio 代替线程
  • React Native移动开发实战-3-实现页面间的数据传递
  • Ruby 2.x 源代码分析:扩展 概述
  • RxJS 实现摩斯密码(Morse) 【内附脑图】
  • Sequelize 中文文档 v4 - Getting started - 入门
  • SpingCloudBus整合RabbitMQ
  • windows下如何用phpstorm同步测试服务器
  • 初识 beanstalkd
  • 基于Volley网络库实现加载多种网络图片(包括GIF动态图片、圆形图片、普通图片)...
  • 理解在java “”i=i++;”所发生的事情
  • 文本多行溢出显示...之最后一行不到行尾的解决
  • 用 Swift 编写面向协议的视图
  • 如何正确理解,内页权重高于首页?
  • ​Java并发新构件之Exchanger
  • #include<初见C语言之指针(5)>
  • $分析了六十多年间100万字的政府工作报告,我看到了这样的变迁
  • (十六)串口UART
  • (未解决)jmeter报错之“请在微信客户端打开链接”
  • (学习日记)2024.02.29:UCOSIII第二节
  • (转)关于pipe()的详细解析
  • (转载)Google Chrome调试JS
  • (轉貼)《OOD启思录》:61条面向对象设计的经验原则 (OO)
  • .Net core 6.0 升8.0
  • .NET Core WebAPI中使用Log4net 日志级别分类并记录到数据库
  • .NET MVC、 WebAPI、 WebService【ws】、NVVM、WCF、Remoting
  • .net企业级架构实战之7——Spring.net整合Asp.net mvc
  • @开发者,一文搞懂什么是 C# 计时器!
  • [ vulhub漏洞复现篇 ] JBOSS AS 5.x/6.x反序列化远程代码执行漏洞CVE-2017-12149
  • [04] Android逐帧动画(一)