实现验证码功能

  1. 先在utils包下创建一个ValidateImageCodeUtils.class

    package com.wfszmg.demo.utils;
    
    import javax.imageio.ImageIO;
    import java.awt.*;
    import java.awt.image.BufferedImage;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.util.Arrays;
    import java.util.Random; /**
    * @author 无法手执玫瑰
    * 2020/07/0021 13:04
    */
    public class ValidateImageCodeUtils {
    /**
    * 验证码难度级别 Simple-数字 Medium-数字和小写字母 Hard-数字和大小写字母
    */
    public enum SecurityCodeLevel {
    Simple, Medium, Hard
    } ; /**
    * 产生默认验证码,4位中等难度
    *
    * @return
    */
    public static String getSecurityCode() {
    return getSecurityCode(4, SecurityCodeLevel.Medium, false);
    } /**
    * 产生长度和难度任意的验证码
    *
    * @param length
    * @param level
    * @param isCanRepeat
    * @return
    */
    public static String getSecurityCode(int length, SecurityCodeLevel level, boolean isCanRepeat) {
    // 随机抽取len个字符
    int len = length;
    // 字符集合(--除去易混淆的数字0,1,字母l,o,O)
    char[] codes = {
    '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
    'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
    'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'
    };
    // 根据不同难度截取字符串
    if (level == SecurityCodeLevel.Simple) {
    codes = Arrays.copyOfRange(codes, 0, 10);
    } else if (level == SecurityCodeLevel.Medium) {
    codes = Arrays.copyOfRange(codes, 0, 36);
    }
    // 字符集和长度
    int n = codes.length;
    // 抛出运行时异常
    if (len > n && isCanRepeat == false) {
    throw new RuntimeException(String.format("调用SecurityCode.getSecurityCode(%1$s,%2$s,%3$s)出现异常," + "当isCanRepeat为%3$s时,传入参数%1$s不能大于%4$s", len, level, isCanRepeat, n));
    }
    // 存放抽取出来的字符
    char[] result = new char[len];
    // 判断能否出现重复字符
    if (isCanRepeat) {
    for (int i = 0; i < result.length; i++) {
    // 索引0 and n-1
    int r = (int) (Math.random() * n);
    // 将result中的第i个元素设置为code[r]存放的数值
    result[i] = codes[r];
    }
    } else {
    for (int i = 0; i < result.length; i++) {
    // 索引0 and n-1
    int r = (int) (Math.random() * n);
    // 将result中的第i个元素设置为code[r]存放的数值
    result[i] = codes[r];
    // 必须确保不会再次抽取到那个字符,这里用数组中最后一个字符改写code[r],并将n-1
    codes[r] = codes[n - 1];
    n--;
    }
    }
    return String.valueOf(result);
    } /**
    * 生成验证码图片
    *
    * @param securityCode
    * @return
    */
    public static BufferedImage createImage(String securityCode) { int codeLength = securityCode.length();//验证码长度 int fontSize = 18;//字体大小 int fontWidth = fontSize + 1; //图片宽高 int width = codeLength * fontWidth + 6;
    int height = fontSize * 2 + 1;
    //图片 BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB); Graphics2D g = image.createGraphics(); g.setColor(Color.WHITE);//设置背景色 g.fillRect(0, 0, width, height);//填充背景 g.setColor(Color.LIGHT_GRAY);//设置边框颜色 g.setFont(new Font("Arial", Font.BOLD, height - 2));//边框字体样式 g.drawRect(0, 0, width - 1, height - 1);//绘制边框 //绘制噪点 Random rand = new Random(); g.setColor(Color.LIGHT_GRAY); for (int i = 0; i < codeLength * 6; i++) { int x = rand.nextInt(width); int y = rand.nextInt(height); g.drawRect(x, y, 1, 1);//绘制1*1大小的矩形 } //绘制验证码 int codeY = height - 10; g.setColor(new Color(19, 148, 246)); g.setFont(new Font("Georgia", Font.BOLD, fontSize));
    for (int i = 0; i < codeLength; i++) {
    double deg = new Random().nextDouble() * 20;
    g.rotate(Math.toRadians(deg), i * 16 + 13, codeY - 7.5);
    g.drawString(String.valueOf(securityCode.charAt(i)), i * 16 + 5, codeY);
    g.rotate(Math.toRadians(-deg), i * 16 + 13, codeY - 7.5);
    } g.dispose();//关闭资源
    return image;
    } public static void main(String[] args) throws IOException {
    String securityCode = ValidateImageCodeUtils.getSecurityCode();
    System.out.println(securityCode); BufferedImage image = ValidateImageCodeUtils.createImage(securityCode);
    ImageIO.write(image, "png", new FileOutputStream("aa.png"));
    } }
  2. 在controller层下创建路由

@GetMapping("/code")
public void getVerification(HttpSession session, HttpServletResponse response) throws IOException {
//生成验证码
String securityCode = ValidateImageCodeUtils.getSecurityCode();
//生成验证码图片
BufferedImage image = ValidateImageCodeUtils.createImage(securityCode);
//存入session
session.setAttribute("code",securityCode);
//响应图片
ServletOutputStream os = response.getOutputStream();
ImageIO.write(image,"png",os);
}

接着访问自己的端口就可以看到验证码

springboot实现验证码功能的更多相关文章

  1. SpringBoot开发验证码功能

    简介 验证码主要是用来防止恶意破解密码.刷票.论坛灌水.刷页.Kaptcha 是一个可高度配置的实用验证码生成工具,使用也很简单,这里就使用它来做验证码. 另外使用JAVA原生的API也可以实现验证码 ...

  2. Vue Springboot (包括后端解决跨域)实现登录验证码功能详细完整版

    利用Hutool 基于Vue.ElementUI.Springboot (跨域)实现登录验证码功能 前言 一.Hutool是什么? 二.下面开始步入正题:使用步骤 1.先引入Hutool依赖 2.控制 ...

  3. Springboot +redis+⾕歌开源Kaptcha实现图片验证码功能

    Springboot +redis+⾕歌开源Kaptcha实现图片验证码功能 背景 注册-登录-修改密码⼀般需要发送验证码,但是容易被 攻击恶意调⽤ 什么是短信-邮箱轰炸机 手机短信轰炸机是批.循环给 ...

  4. JavaFX+SpringBoot+验证码功能的小型薪酬管理系统

    2020.07.22更新 1 概述 1.1 简介 一个简单的小型薪酬管理系统,前端JavaFX+后端Spring Boot,功能倒没多少,主要精力放在了UI和前端的一些逻辑上面,后端其实做得很简单. ...

  5. Springboot 生成验证码

    技术:springboot+kaptcha+session   概述 场景介绍 验证码,用于web网站.用户点击验证码图片后,生成验证码.提交后,用户输入验证码和Session验证码,进行校验. 详细 ...

  6. 一步一步实现web程序信息管理系统之三----登陆业务逻辑实现(验证码功能+参数获取)

    本篇紧接着上一篇文章[一步一步实现web程序信息管理系统之二----后台框架实现跳转登陆页面] 验证码功能 一般验证码功能实现方式为,前端界面访问一个url请求,后端服务代码生成一个图片流返回至浏览器 ...

  7. dd——留言板再加验证码功能

    1.找到后台-核心-频道模型-自定义表单 2.然后点击增加新的自定义表单 diyid 这个,不管他,默认就好 自定义表单名称 这个的话,比如你要加个留言板还是投诉建议?写上去呗 数据表  这个不要碰, ...

  8. .Net Core 之 图形验证码 本文介绍.Net Core下用第三方ZKWeb.System.Drawing实现验证码功能。

    本文介绍.Net Core下用第三方ZKWeb.System.Drawing实现验证码功能. 通过测试的系统: Windows 8.1 64bit Ubuntu Server 16.04 LTS 64 ...

  9. c#实现验证码功能

    一.验证码简介 验证码功能一般是用于防止批量注册的,不少网站为了防止用户利用机器人自动注册.登录.灌水,都采用了验证码技术.所谓验证码,就是将一串随机产生的数字或字母或符号或文字,生成一幅图片, 图片 ...

  10. javaweb实现验证码功能

    在javaweb的用户注册与登陆功能时,有时为了防止漏洞或者大量注册,可以使用验证码功能,下面是验证码的一个简单实现 验证码类 public class ValiImg extends HttpSer ...

随机推荐

  1. Vue2路由跳转传参,获取路由参数,Vue监听路由

    1 this.$router.push({ 2 // name:路由组件名 3 name: routeName, 4 query: { 5 mapId:this.mapId 6 } 7 }) 8 9 ...

  2. mysql锁表原因及解决方法

    mysql锁表原因及解决方法   一.导致锁表的原因 1.锁表发生在insert update .delete 中: 2.锁表的原理是 数据库使用独占式封锁机制,当执行上面的语句时,对表进行锁住,直到 ...

  3. Charles 抓取 HTTPS 协议内容,需要做什么操作?

    抓取 HTTPS 需要安装证书,Charles 端需要安装 Android.iOS手机端也需要安装 电脑的 Charles 操作:1.proxy - proxy setting - http prox ...

  4. Vue声明式渲染、条件与循环、事件绑定、双向绑定及生命周期钩子函数

    VUE基础介绍 Vue 是一套用于构建用户界面的渐进式框架.与其它大型框架不同的是,Vue 被设计为可以自底向上逐层应用. -声明式渲染 <div> {{ message }} </ ...

  5. java中锁的应用(ReentrantLock)

    package com.xielu.test; public class explicit { private static Lock lock = new ReentrantLock(); priv ...

  6. 需要改动node_modules,并且别人也可以同步,插件 patch-package

    patch-package 转自:https://www.cnblogs.com/lovewhatIlove/p/15724812.html 1.简介:有个功能需要修改node_modules里面的代 ...

  7. VOIP(SIP)呼叫环境及流程试验

    宿主机:win11  IP: .1         PHONE: 102 虚拟机: v11     IP: .129     SIP SERVER 虚拟机: v10     IP: .128      ...

  8. 匿名Lambda函数,C++

    1 // To Compile and Run: g++ -std=c++11 lambda.cc -Wall -O3 && ./a.out 2 3 4 #include <io ...

  9. 如何跳出forEach循环

    for(let ii in this.listData){ console.log("提交前数据",ii) try{ this.listData[ii].forEach((el,i ...

  10. mysql和nacos都部署在docker中,ip该写哪个

    docker run -d \ -e MODE=standalone \ -e SPRING_DATASOURCE_PLATFORM=mysql \ -e MYSQL_SERVICE_HOST=172 ...