实现验证码功能

  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. 关于unity游戏的类名查找

    这里我用火影演示 首先我们知道,鸣人的决斗场技能代码为900010151,0251,0351,0451等等 我们进行搜索 然后再转到第二个进去(第一个为忍者技能的序列表) 注:序列表里的代码与那个他相 ...

  2. IOS弹出系统键盘后,页面不恢复

    <script> var u = navigator.userAgent, app = navigator.appVersion var isIOS = !!u.match(/\(i[^; ...

  3. epoll 基本知识与使用

    https://blog.csdn.net/qq_35721743/article/details/86742508 epoll 最大的好处在于它不会随着监听 fd 数目的增长而降低效率. epoll ...

  4. 全局监控Promise错误

    一.问题引入 Promise 在前端中的使用已经非常普遍了,但是许多开发者或许习惯了链式调用却忘了捕获 Promise 的错误了. 例如: function forgetCatchError () { ...

  5. python基于word模板批量生成word文件

    1.需要用到docxtpl库,用于操作word模板 安装:pip insatll docxtpl 处理之前的word模板 处理后的word 下面直接上代码揭开它的神秘面纱:第一步,读取excel中的内 ...

  6. 【项目记录】3:pyinstaller打包之后tkcalendar无法使用的解决办法

    PyInstaller 没有检测到二级导入这一事实.tkcalendar 的HowTos文档中解释了解决此问题的方法: 使用 PyInstaller 捆绑应用程序时, 检测 tkcalendar 的 ...

  7. Nginx 监听同一端口号配置多个域名

    同一台nginx服务器通过配置多个server块实现在同一端口号下监听多个域名. 需要注意的是:端口号(listen)+主机名(server_name) 需要在多个server中唯一,否则会报错. 实 ...

  8. 12.8 linux学习第十五天

    今天老刘讲了第11章和第12章,感觉讲的速度很快,一气呵成,水都没怎么喝. 11.1 文件传输协议 一般来讲,人们将计算机联网的首要目的就是获取资料,而文件传输是一种非常重要的获取资料的方式.今天的互 ...

  9. window10下,命令行与端口

    netstat -ano 查看端口情况 tasklist|findstr "9220" 通过PID号"9220"查看对应端口被什么进程占用了 netstat - ...

  10. C++解决海盗分金问题

    #include <iostream> #include <unordered_map> #include <map> // 假定:每个海盗都是一样的聪明,没有谁比 ...