java 自定义生成验证码

news2025/8/1 19:20:35

目录

  • 说明
  • 效果展示
  • Base64编码的文件类型枚举类
  • 验证码默认常量值
  • 验证码生成工具类
  • 使用

说明

项目登录或者其他重要的操作中都要生成验证码,其重要性在此不多说。
主要是介绍自己封装的验证码生成工具类的使用。

  1. 建议安装lombok插件,不使用此插件则需要手动生成get、set方法
  2. 此工具类不需要引入第三方jar依赖。
  3. 验证码说明:
    验证码由数字和字母组成
    支持自定义验证码位数
    支持自定义验证码生成规则
    支持自定义验证码图片宽高
    支持验证码生成图片保存到指定路径
    支持验证码转成base64编码
    支持自定义验证码文件和base64编码前缀格式

效果展示

在这里插入图片描述

Base64编码的文件类型枚举类

import lombok.Getter;

@Getter
public enum Base64FileTypeEnum {
    // 文件类型
    BASE64_FILETYPE_DOC(".doc", "data:application/msword;base64"),
    BASE64_FILETYPE_DOCX(".docx", "data:application/vnd.openxmlformats-officedocument.wordprocessingml.document;base64"),
    BASE64_FILETYPE_XLS(".xls", "data:application/vnd.ms-excel;base64"),
    BASE64_FILETYPE_XLSX(".xlsx", "data:application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;base64"),
    BASE64_FILETYPE_PDF(".pdf", "data:application/pdf;base64"),
    BASE64_FILETYPE_PPT(".ppt", "data:application/vnd.ms-powerpoint;base64"),
    BASE64_FILETYPE_PPTX(".pptx", "data:application/vnd.openxmlformats-officedocument.presentationml.presentation;base64"),
    BASE64_FILETYPE_TXT(".txt", "data:text/plain;base64"),

    // 图片类型
    BASE64_FILETYPE_PNG(".png", "data:image/png;base64"),
    BASE64_FILETYPE_JPG(".jpg", "data:image/jpeg;base64"),
    BASE64_FILETYPE_JPEG(".jpeg", "data:image/jpeg;base64"),
    BASE64_FILETYPE_GIF(".gif", "data:image/gif;base64"),
    BASE64_FILETYPE_SVG(".svg", "data:image/svg+xml;base64"),
    BASE64_FILETYPE_ICO(".ico", "data:image/x-icon;base64"),
    BASE64_FILETYPE_BMP(".bmp", "data:image/bmp;base64");

    private String code;
    private String value;

    private Base64FileTypeEnum(String code, String value) {
        this.code = code;
        this.value = value;
    }


}

验证码默认常量值


/**
 * 说明: 验证码常量
 *
 * @param
 * @author zhangxiaosan
 * @create 2022/11/25
 * @return
 */
public interface ValidateCodeConstant {
    /**
     * 说明: 默认验证码长度
     *
     * @author zhangxiaosan
     * @create 2022/11/25
     * @param
     * @return
     */
    Integer size = 4;

    /**
     * 说明: 验证码图片宽
     *
     * @author zhangxiaosan
     * @create 2022/11/25
     * @param
     * @return
     */
    Integer width = 120;

    /**
     * 说明: 验证码图片高
     *
     * @author zhangxiaosan
     * @create 2022/11/25
     * @param
     * @return
     */
    Integer height = 40;

    /**
    * 说明: 生成的图片格式,默认jpg
    * @author   zhangxiaosan
    * @create   2022/11/25
    * @param
    * @return
    */
    String imgType = Base64FileTypeEnum.BASE64_FILETYPE_JPG.getCode().replaceAll("\\.","");

    /**
     * 说明: 生成的base64前缀,默认jpg
     * @author   zhangxiaosan
     * @create   2022/11/25
     * @param
     * @return
     */
    String base64Prefix = Base64FileTypeEnum.BASE64_FILETYPE_JPG.getValue() + ",";

    /**
    * 说明: 验证码生成图片存储地址,默认不存放图片
    * @author   zhangxiaosan
    * @create   2022/11/25
    * @param
    * @return
    */
    String saveImagePath = null;
}

验证码生成工具类


import org.springframework.util.StringUtils;
import sun.misc.BASE64Encoder;
import www.three.commom.constant.ValidateCodeConstant;
import www.three.commom.enums.Base64FileTypeEnum;

import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.io.*;
import java.util.*;

/**
 * 说明:
 * 生成验证码工具类
 *
 * @author 张小三
 * @create 2022-11-24 22:48
 * @verson 1.0.0
 */
public class ValidateCodeUtil {
    private static Random random = new Random();

    private static Color getRandColor(int fc, int bc) {
        if (fc > 255)
            fc = 255;
        if (bc > 255)
            bc = 255;
        int r = fc + random.nextInt(bc - fc);
        int g = fc + random.nextInt(bc - fc);
        int b = fc + random.nextInt(bc - fc);
        return new Color(r, g, b);
    }

    private static int getRandomIntColor() {
        int[] rgb = getRandomRgb();
        int color = 0;
        for (int c : rgb) {
            color = color << 8;
            color = color | c;
        }
        return color;
    }

    private static int[] getRandomRgb() {
        int[] rgb = new int[3];
        for (int i = 0; i < 3; i++) {
            rgb[i] = random.nextInt(255);
        }
        return rgb;
    }

    private static void shear(Graphics g, int w1, int h1, Color color) {
        shearX(g, w1, h1, color);
        shearY(g, w1, h1, color);
    }

    private static void shearX(Graphics g, int w1, int h1, Color color) {

        int period = random.nextInt(2);

        boolean borderGap = true;
        int frames = 1;
        int phase = random.nextInt(2);

        for (int i = 0; i < h1; i++) {
            double d = (double) (period >> 1)
                    * Math.sin((double) i / (double) period + (6.2831853071795862D * (double) phase) / (double) frames);
            g.copyArea(0, i, w1, 1, (int) d, 0);
            if (borderGap) {
                g.setColor(color);
                g.drawLine((int) d, i, 0, i);
                g.drawLine((int) d + w1, i, w1, i);
            }
        }

    }

    private static void shearY(Graphics g, int w1, int h1, Color color) {

        int period = random.nextInt(40) + 10; // 50;

        boolean borderGap = true;
        int frames = 20;
        int phase = 7;
        for (int i = 0; i < w1; i++) {
            double d = (double) (period >> 1)
                    * Math.sin((double) i / (double) period + (6.2831853071795862D * (double) phase) / (double) frames);
            g.copyArea(i, 0, 1, h1, 0, (int) d);
            if (borderGap) {
                g.setColor(color);
                g.drawLine(i, (int) d, i, 0);
                g.drawLine(i, (int) d + h1, i, h1);
            }

        }
    }

    /**
     * 生成指定长度的随机数字和字母
     *
     * @param length
     * @return
     */
    private static String getStringRandom(int length) {
        String val = "";
        Random random = new Random();
        for (int i = 0; i < length; i++) {
            String charOrNum = random.nextInt(2) % 2 == 0 ? "char" : "num";
            switch (charOrNum) {
                case "char":
                    int temp = random.nextInt(2) % 2 == 0 ? 65 : 97;
                    val += (char) (random.nextInt(26) + temp);
                    break;
                case "num":
                    val += String.valueOf(random.nextInt(10));
                    break;
            }
        }
        return val;
    }

    /**
     * 创建Base64编码的验证码图片
     *
     * @param w    宽
     * @param h    高
     * @param code 验证码
     * @return
     * @throws Exception
     */
    private static ByteArrayOutputStream imageToBase64Byte(int w, int h, String code) throws Exception {
        int verifySize = code.length();
        BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
        Random rand = new Random();
        Graphics2D g2 = image.createGraphics();
        g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        Color[] colors = new Color[5];
        Color[] colorSpaces = new Color[]{Color.WHITE, Color.CYAN, Color.GRAY, Color.LIGHT_GRAY, Color.MAGENTA,
                Color.ORANGE, Color.PINK, Color.YELLOW};
        float[] fractions = new float[colors.length];
        for (int i = 0; i < colors.length; i++) {
            colors[i] = colorSpaces[rand.nextInt(colorSpaces.length)];
            fractions[i] = rand.nextFloat();
        }
        Arrays.sort(fractions);

        g2.setColor(Color.GRAY);// 设置边框色
        g2.fillRect(0, 0, w, h);

        Color c = getRandColor(200, 250);
        g2.setColor(c);// 设置背景色
        g2.fillRect(0, 2, w, h - 4);

        // 绘制干扰线
        Random random = new Random();
        g2.setColor(getRandColor(160, 200));// 设置线条的颜色
        for (int i = 0; i < 20; i++) {
            int x = random.nextInt(w - 1);
            int y = random.nextInt(h - 1);
            int xl = random.nextInt(6) + 1;
            int yl = random.nextInt(12) + 1;
            g2.drawLine(x, y, x + xl + 40, y + yl + 20);
        }

        // 添加噪点
        float yawpRate = 0.05f;// 噪声率
        int area = (int) (yawpRate * w * h);
        for (int i = 0; i < area; i++) {
            int x = random.nextInt(w);
            int y = random.nextInt(h);
            int rgb = getRandomIntColor();
            image.setRGB(x, y, rgb);
        }

        shear(g2, w, h, c);// 使图片扭曲

        g2.setColor(getRandColor(100, 160));
        int fontSize = h - 4;
        Font font = new Font("Arial", Font.ITALIC, fontSize);
        g2.setFont(font);
        char[] chars = code.toCharArray();
        for (int i = 0; i < verifySize; i++) {
            AffineTransform affine = new AffineTransform();
            affine.setToRotation(Math.PI / 4 * rand.nextDouble() * (rand.nextBoolean() ? 1 : -1),
                    (w / verifySize) * i + fontSize / 2, h / 2);
            g2.setTransform(affine);
            g2.drawChars(chars, i, 1, ((w - 10) / verifySize) * i + 5, h / 2 + fontSize / 2 - 10);
        }
        g2.dispose();
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        ImageIO.write(image, ValidateCodeConstant.imgType, baos);
        return baos;
        //return new Base64().encodeToString(baos.toByteArray());
        //return new BASE64Encoder().encode(baos.toByteArray());
        //return Base64.getEncoder().encodeToString(baos.toByteArray());
    }


    /**
     * 创建Base64编码的验证码图片,并保存到指定的路径
     *
     * @param w        宽
     * @param h        高
     * @param code     验证码
     * @param savePath 验证码图片保存的地址
     * @return String 单行的base64编码,携带文件前缀
     * @throws Exception
     */
    private static String createValidateCodeImageBase(Integer w, Integer h, String code, String savePath) throws Exception {
        ByteArrayOutputStream baos = imageToBase64Byte(w, h, code);
        if (StringUtils.hasText(savePath)) {
            try {
                File file = new File(savePath);
                FileOutputStream write = new FileOutputStream(file);
                byte[] decode = Base64.getEncoder().encode(new BASE64Encoder().encode(baos.toByteArray()).getBytes());
                write.write(decode);
                write.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        // 返回不换行的编码
        return ValidateCodeConstant.base64Prefix + Base64.getEncoder().encodeToString(baos.toByteArray());
    }

    /**
     * 说明: 生成自定义验证码 图片大小
     * 验证码宽为 120,高为40,长度为4位,不生成为图片存放
     *
     * @param
     * @return String 单行的base64编码,携带文件前缀
     * @author zhangxiaosan
     * @create 2022/11/24
     */
    public static Map<String, Object> createValidateCodeImage(Integer w, Integer h, String code, String savePath) throws Exception {
        String validateCodeImage = createValidateCodeImageBase(w, h, code, savePath);
        Map<String, Object> res = new HashMap<String, Object>();
        res.put("code", code);
        res.put("base64", validateCodeImage);
        res.put("w", w);
        res.put("h", h);
        res.put("uuid",UUID.randomUUID());
        return res;
    }


    /**
     * 说明: 生成默认的验证码
     * 验证码宽为 120,高为40,长度为4位,不生成为图片存放
     *
     * @param
     * @return String 单行的base64编码,携带文件前缀
     * @author zhangxiaosan
     * @create 2022/11/24
     */
    public static Map<String, Object> createValidateCodeImage() throws Exception {
        String code = getStringRandom(ValidateCodeConstant.size);
        return createValidateCodeImage( ValidateCodeConstant.width, ValidateCodeConstant.height, code, ValidateCodeConstant.saveImagePath);
    }

    /**
     * 说明: 生成自定义验证码位数
     * 验证码宽为 120,高为40,长度为4位,不生成为图片存放
     *
     * @param
     * @return String 单行的base64编码,携带文件前缀
     * @author zhangxiaosan
     * @create 2022/11/24
     */
    public static Map<String, Object> createValidateCodeImage(Integer size) throws Exception {
        String code = getStringRandom(size);
        return createValidateCodeImage(ValidateCodeConstant.width, ValidateCodeConstant.height, code, ValidateCodeConstant.saveImagePath);
    }

    /**
     * 说明: 生成自定义验证码 图片大小
     * 长度为4位,不生成为图片存放
     *
     * @param
     * @return String 单行的base64编码,携带文件前缀
     * @author zhangxiaosan
     * @create 2022/11/24
     */
    public static Map<String, Object> createValidateCodeImage(Integer w, Integer h) throws Exception {
        String code = getStringRandom(ValidateCodeConstant.size);
        return createValidateCodeImage(ValidateCodeConstant.width, ValidateCodeConstant.height, code, ValidateCodeConstant.saveImagePath);
    }

    /**
     * 说明: 生成自定义验证码 图片大小
     * 长度为4位,不生成为图片存放
     *
     * @param
     * @return String 单行的base64编码,携带文件前缀
     * @author zhangxiaosan
     * @create 2022/11/24
     */
    public static Map<String, Object> createValidateCodeImage(Integer w, Integer h,Integer size) throws Exception {
        String code = getStringRandom(size);
        return createValidateCodeImage( w,  h, code, ValidateCodeConstant.saveImagePath);
    }

    
}

使用

public static void main(String[] args) throws Exception {
     ValidateCodeUtil.createValidateCodeImage().forEach((k,v)->{
         System.out.println(k+":"+v);
     });
 }

结果如下:

code: mo1U
w: 120
base64: 
h: 40
uuid: 404d2e57-f666-4958-9557-2c113af654bf

结果说明:
w : 宽
h :高
code: 验证码
uuid: 验证码唯一标识
base64:验证码图片编码,可以传给前端在html 的 <img src=‘’ "/ > 中展示验证码图片

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/35087.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

cron表达式,结构、字段说明、特殊字符说明、常用表达式

1.cron表达式的结构 Cron表达式是一个字符串&#xff0c;结构非常简单。Cron表达式从左到右分为6或7个字段&#xff0c;每个字段代表一个含义&#xff0c;用空格隔开。如下图所示&#xff1a; 2.cron表达式中各个字段的说明和规则 Cron一共有7位&#xff0c;最后一位是年份&…

浅析资源调度框架YARN

第一章 资源调度框架YARN理论 1.1 YARN概述 分布式操作系统 hadoop 1.xMapReduce主从架构 主节点JobTracker 从节点TaskTrackerslot hadoop 2.xMapReduce编程API YARN主从架构 主节点ResourceManager 从节点NodeManagerContainer hadoop 3.xCommonHDFS 纠删码 …

关于升级高德地图导航9.5.0的问题 ‘com.amap.api:navi-3dmap:9.5.0_3dmap9.5.0‘

最近打开项目&#xff0c;发现高德有新版本更新&#xff0c;果断更新。哈哈哈哈。然而结果好像并没有这么简单。要是世界上什么事情这么简单就好了。年轻人。还是太年轻了啊。 然后更新完最新的依赖 /*高德地图远程依赖*/implementation com.amap.api:navi-3dmap:9.5.0_3dmap9…

I/O 设备(输入/输出设备)

文章目录I/O 设备输入设备输出设备1&#xff0c;显示器2&#xff0c;打印机3&#xff0c;投影仪I/O 设备 输入设备 借助计算机的输入设备&#xff0c;用户能够轻松地将数据或者指令传递给计算机。同时&#xff0c;计算机中的 CPU 会接收用户输入的指令或数据&#xff0c;并对…

Tesla M40 下Ubuntu anaconda pycharm pytorch安装

显卡&#xff1a;Tesla M40 24GB (2张&#xff09; 显卡驱动版本(推荐)&#xff1a;470.57.02 cuda版本&#xff1a;11.4 安装前需要&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;禁用nouveau驱动&#xff01;&#xff01;&#xff01;&#xff01;&#x…

矢量网络分析仪的S11和S12参数是什么呢?

矢量网络分析仪的基础功能是S参数测试。所谓S参数&#xff0c;就是散射参数&#xff0c;是描述电磁波在被测设备的入射波量、反射波量以及波量传输特性的参数。简单理解&#xff1a;S11代表端口1的反射&#xff0c;S22代表端口2的反射&#xff0c;S21是端口1至端口2的传输&…

阿里云安装软件:jdk11

命令下载 1. 安装准备 检查系统jdk版本 java -version检查jdk自带安装包 rpm -qa | grep java卸载jdk yum -y remove tzdata-java.noarch如果有就卸载&#xff0c;卸载的包名通过&#xff08;rpm -qa | grep java&#xff09;获取&#xff0c;包名要全部输入 rpm -e --nodeps …

Kotlin基础认知 - 为何Kotlin文件有的带.kt后缀,有的不带?

有一天看到项目中的Kotlin类&#xff0c;有的有.kt后缀&#xff0c;有的没有&#xff0c;针对这个情况我就简单看了下&#xff0c;然后记录一波 创建 Kotlin Class 或 Kotlin File创建Kotlin class创建Kotlin File俩者区别展现形式外部展现内部展现延伸扩展、对向转换Class无后…

【Silvaco example】Temperature Ramping - Effect on Leakage

1、例子讲解 本示例演示了Atlas中任何device的全局温度梯度&#xff08;global temperature ramping&#xff09;的正确方法。 &#xff08;1&#xff09;结构定义 为了简单起见&#xff0c;这里选择了二极管结构。 go atlasmeshx.mesh loc0.00 spac0.05 x.mesh loc0.10 sp…

简单手段发IF=7+文章:磷酸三苯酯对鲤鱼的毒性作用及肠道微生物群落影响

研究背景 磷酸三苯基酯&#xff08;TPHP&#xff09;是一种有机磷阻燃剂&#xff0c;它通过挥发以及溶解分散到环境中&#xff0c;并通过食物链富集生物体&#xff0c;对生态系统产生不可避免的负面影响。已发现TPHP可以引起组织病变&#xff0c;干扰脂质代谢&#xff0c;并降…

全新营销时代,金融企业如何有“种”有“收”?

贯穿2022年的主题&#xff0c;就是“不确定性”。 复杂的大环境下&#xff0c;金融行业的发展饱受“震荡”。疫情、通胀&#xff0c;乃至二级市场的风云变幻&#xff0c;都在考验金融企业经营的确定性。那么&#xff0c;金融企业踏平波动、坚定前行的力量从何而来&#xff1f;…

Spring Boot 简介及快速搭建

Spring Boot 简介及快速搭建 springboot的优点&#xff1a; –快速构建一个独立的 Spring 应用程序 &#xff1b; –嵌入的 Tomcat 、 Jetty 或者 Undertow&#xff0c;无须部署 WAR 文件&#xff1b; –提供starter POMs来简化Maven配置和减少版本冲突所带来的问题&#xff1…

Vue3 setup函数的使用

全新的 setup 函数 在开始编写 Vue 组件之前&#xff0c;需要了解两个全新的前置知识点&#xff1a; 全新的 setup 函数&#xff0c;关系到组件的生命周期和渲染等问题 写 TypeScript 组件离不开的 defineComponent API setup 的含义 Vue 3 的 Composition API 系列里&#x…

计算机网络---应用层概述

&#xff08;一&#xff09;应用层概述 基本定义&#xff1a; 应用层&#xff08;Application layer&#xff09;是OSI模型的第七层。应用层直接和应用程序接口并提供常见的网络应用服务。应用层也向表示层发出请求。应用层是开放系统的最高层,是直接为应用进程提供服务的。其…

数据可视化设计经验分享:10分钟做出炫酷数据大屏

又快到年终了&#xff0c;数据大屏作为一个数据管理的分析工具&#xff0c;在年底数据大屏的制作需求日益增加。在一些公司比如银行、证券、医院、外贸等“数据大户”&#xff0c;在数据大屏的应用上更是有大量需求。 在接下来的两个月&#xff0c;许多人会感到痛苦吧&#xf…

.net---继承和多态

继承和多态继承和多态的基本概念继承继承的类型实现继承接口继承继承派生类base关键字构造函数的调用类成员的继承类成员的隐藏继承&#xff1a;虚方法和隐藏方法抽象类和抽象方法抽象类抽象方法密封类和密封方法密封类接口接口成员接口实现多态重载重写继承和多态的基本概念 …

NIO中ByteBuffer

// Invariants: mark < position < limit < capacityprivate int mark -1;private int position 0;private int limit;private int capacity; 从源码中可以看出&#xff0c;ByteBuffer的几个实例变量。我们稍后会详细解析这几个变量的意义。 在我们刚创建ByteBuff…

绿色积分消费时代来临,共享购跟随国家的号召

近年来&#xff0c;中国共享经济商业模式不断发展成熟。自2016年国家“十三五”规划纲要首次提出“共享经济”概念以来&#xff0c;中国政府不断助推共享经济发展。最新发布的《共享经济指导原则与基本框架》为共享经济作出定义&#xff1a;共享经济即“资源供给者通过平台与资…

IDEA配置Maven

1. IDEA配置Maven环境 先在IDEA中配置Maven环境&#xff1a; 选择 IDEA中 File --> Settings 搜索 maven 设置 IDEA 使用本地安装的 Maven&#xff0c;并修改配置文件路径 2. IDEA 创建 Maven项目 创建模块 创建模块&#xff0c;选择Maven&#xff0c;点击Next 填写模块名称…

你不知道的Spring的依赖的查找和注入的来源

1.写在前面 前面的博客我们已经介绍完了spring的依赖的查找和注入的方式&#xff0c;这篇博客我们主要介绍下spring的依赖的查找和注入的来源。 2.依赖查找的来源 查找来源 Spring 內建 BeanDefintion Spring 內建单例对象 上面的各种依赖都是在spring的生命周期的过程中&am…