自己实现图形验证码

如果不想重复造轮子,参考上一篇文章:SpringBoot生成图形验证码_Muscleheng的博客-CSDN博客

这里不需要依赖开源组件包,完全自己实现图形验证码功能

两步完成:

第一步:编写图形验证码工具

package com.zhh.demo.common.util;import lombok.extern.slf4j.Slf4j;
import org.apache.tomcat.util.codec.binary.Base64;
import org.springframework.stereotype.Component;import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.Random;/*** @Description: 图形验证码* @Author: zhaoheng* @CreateTime: 2022-12-09*/
@Slf4j
@Component
public class GraphValidateCode {//设置图片宽private int width = 70;//设置图片高度private int height = 30;//设置干扰线数量private int lineSize = 40;/** 随机产生数字和字母组合的字符串,字体只显示大写,去掉了1,0,i,o几个容易混淆的字符 */public static final String VERIFY_CODES = "23456789ABCDEFGHJKLMNPQRSTUVWXYZ";/*** 获得字体*/private Font getFont() {return new Font("Fixedsys", Font.CENTER_BASELINE, 18);}/*** 获得颜色*/private Color getRandColor(int fc, int bc) {Random random = new Random();if (fc > 255) {fc = 255;}if (bc > 255) {bc = 255;}int r = fc + random.nextInt(bc - fc - 16);int g = fc + random.nextInt(bc - fc - 14);int b = fc + random.nextInt(bc - fc - 18);return new Color(r, g, b);}/*** 获取验证码** @return*/public String getIdentifyCode() {Random random = new Random();StringBuffer buffer = new StringBuffer();for (int i = 0; i < 4; i++) {char c = VERIFY_CODES.charAt(random.nextInt(VERIFY_CODES.length()));buffer.append(c);}return buffer.toString();}/*** 生成随机图片** @param identifyCode  图形码值* @return*/public BufferedImage getIdentifyImage(String identifyCode) {//BufferedImage类是具有缓冲区的Image类,Image类是用来描述图像信息的类BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_BGR);//产生Image对象的Graphics对象,改对象可以在图像上进行各种绘制操作Graphics graphics = image.getGraphics();//图片大小graphics.fillRect(0, 0, width, height);//字体大小graphics.setFont(new Font("Times New Roman", Font.ROMAN_BASELINE, 20));//字体颜色graphics.setColor(getRandColor(110, 133));//绘制干扰线for (int i = 0; i <= lineSize; i++) {drawLine(graphics);}//绘制随机字符drawString(graphics, identifyCode);graphics.dispose();return image;}/*** 绘制字符串* @param g     产生Image对象的Graphics对象* @param identifyCode  图形码值*/private void drawString(Graphics g, String identifyCode) {Random random = new Random();for (int i = 0; i < identifyCode.length(); i++) {g.setFont(getFont());g.setColor(new Color(random.nextInt(101), random.nextInt(111), random.nextInt(121)));g.translate(random.nextInt(3), random.nextInt(3));// x:图形码值基于图片最左边的距离g.drawString(String.valueOf(identifyCode.charAt(i)), 13 * i + 8, 18);}}/*** 绘制干扰线*/private void drawLine(Graphics graphics) {Random random = new Random();int x = random.nextInt(width);int y = random.nextInt(height);int xl = random.nextInt(13);int yl = random.nextInt(15);graphics.drawLine(x, y, x + xl, y + yl);}/*** 响应验证码图片** @param identifyImg   图形码对象* @param response*/public void responseIdentifyImg(BufferedImage identifyImg, HttpServletResponse response) {//设置响应类型,告诉浏览器输出的内容是图片response.setContentType("image/jpeg");//设置响应头信息,告诉浏览器不用缓冲此内容response.setHeader("Pragma", "No-cache");response.setHeader("Cache-Control", "no-cache");response.setDateHeader("Expire", 0);try {//把内存中的图片通过流动形式输出到客户端ImageIO.write(identifyImg, "JPEG", response.getOutputStream());} catch (IOException e) {log.error("图形验证码输出错误", e);}}/*** BufferedImage转base64* @param img  BufferedImage 对象* @return*/public String getBase64FromImage(BufferedImage img) {String base64 = "";try (ByteArrayOutputStream stream = new ByteArrayOutputStream()) {// 设置图片的格式ImageIO.write(img, "jpg", stream);byte[] bytes = Base64.encodeBase64(stream.toByteArray());base64 = new String(bytes);} catch (IOException e) {log.error("图形验证码转base64错误", e);}return "data:image/jpeg;base64," + base64;}}

第二步:编写接口

两个接口,两种方式:1.接口直接返回图片。2.接口返回base64字符串。

@Api(tags = "验证码-api")
@RequestMapping("/api/test")
@RestController
public class TestController {// 模拟把验证码的值存储到缓存(记得添加过期时间)Map<String,String> hashMap = new HashMap<>();@Autowiredprivate GraphValidateCode graphValidateCode;/*** 给前端返回一个验证码图片* @return*/@ApiOperation("获取图形验证码")@GetMapping("/identifyImage")public void identifyImage(HttpServletResponse response,@ApiParam(value = "图形验证码id,无值:生成验证码,有值:刷新验证码")@RequestParam(name = "codeId", required = false) String codeId) {// 验证码String identifyCode = graphValidateCode.getIdentifyCode();// 模拟把验证码的值存储到缓存(记得添加过期时间)if (codeId == null) {System.out.println("获取图形码");codeId = ToolUtil.simpleUUID();// 保存图形码值hashMap.put(codeId, identifyCode);} else {System.out.println("刷新图形码");// 更新图形码值,此时此刻 图形码可能已经过期删除,那就相对于保存一个新的hashMap.put(codeId, identifyCode);}// 图形验证码对应的UUID放在header中,前端可以拿到response.setHeader("codeId", codeId);//根据验证码创建图片BufferedImage identifyImage = graphValidateCode.getIdentifyImage(identifyCode);//回传给前端graphValidateCode.responseIdentifyImg(identifyImage, response);}/*** 给前端返回一个验证码图片-base64格式* @return*/@ApiOperation("获取图形验证码-base64格式")@GetMapping("/identifyImage2")public String identifyImage2(@ApiParam(value = "图形验证码id,无值:生成验证码,有值:刷新验证码")@RequestParam(name = "codeId", required = false) String codeId) {String identifyCode = graphValidateCode.getIdentifyCode();// 模拟把验证码的值存储到缓存(记得添加过期时间)if (codeId == null) {System.out.println("获取图形码");codeId = ToolUtil.simpleUUID();// 保存图形码值hashMap.put(codeId, identifyCode);} else {System.out.println("刷新图形码");// 更新图形码值,此时此刻 图形码可能已经过期删除,那就相对于保存一个新的hashMap.put(codeId, identifyCode);}//根据验证码创建图片BufferedImage identifyImage = graphValidateCode.getIdentifyImage(identifyCode);//回传给前端System.out.println("新图形码值:" + identifyCode);return graphValidateCode.getBase64FromImage(identifyImage);}}

效果展示:

 

 

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

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

相关文章

图形验证码的使用

在用户登录的时候&#xff0c;除了要输入用户名和密码&#xff0c;有时候还需要输入验证码进行验证&#xff0c;如下&#xff1a; 现在一般用短信验证码比较多&#xff0c;但是图形验证码也有使用。记录一下图形验证码的使用过程。 1、验证码生成器 先定义一个验证码的生成器…

vue图形验证码

Vue图形验证码 组件 可以自行封装一下&#xff0c;放在components目录下。 <template><span class"s-canvas" click"changeCode"><canvas id"s-canvas" :width"contentWidth" :height"contentHeight">…

图形验证码安全

目录 图形验证码 图形验证码的作用和原理 图形验证码的分类 图形验证码的验证过程 图形验证码的安全问题 静态图形验证码的破解 利用Python脚本破解静态图形验证码 图形验证码 我们经常在登录app或者网页的时候&#xff0c;都会需要我们输入图形验证码上的内容&#xf…

图形验证码

一、图形验证码是什么&#xff1f; 图形验证码是一些没有规则的图文的组合&#xff0c;参考下图 二、图形验证码有什么用&#xff1f; 防止恶意攻击者采用恶意工具批量注册账号或是大量频繁调用某些请求&#xff0c;给服务器造成压力&#xff0c;占用大量的系统资源。 三、图形…

C语言——数据在内存中的存储(上)

数据在内存中的存储 1. 数据类型的介绍 之前已经介绍过C语言中的基本数据类型了&#xff0c;主要有&#xff1a; char //字符数据类型short //短整型int //整形long //长整型long long //更长的整形float //单精度浮点数double //双精度浮点数 注意&#xff1a;C语言中是是没…

php图形 验证码代码,PHP制作图形验证码代码分享,php图形验证码代码_PHP教程

PHP制作图形验证码代码分享,php图形验证码代码 效果: myvcode.class.php:封装创建验证码的类 /* * file:myvcode.class.php * 验证码类,类名Vcode */ class Vcode {private $width; /*验证码宽度*/ private $height; /*验证码高度*/ private $codeNum; /*验证码字符个数*/ p…

Android图形验证码

1. 前言 图形验证码可以让服务器以图片的形式传给客户端&#xff0c;也可以让客户端自己实现。那客户端要怎么做呢&#xff1f;其实很简单&#xff0c;可以使用Android的Canvas、Paint和Random来实现。用Random来随机生成数字、字母、颜色、画笔原点等等&#xff0c;设置Paint…

java生成图形验证码

随时随地阅读更多技术实战干货&#xff0c;获取项目源码、学习资料&#xff0c;请关注源代码社区公众号(ydmsq666) 首先&#xff0c;需要生成验证码字符串&#xff0c;方式很多&#xff0c;下面提供一种&#xff0c;根据指定源的方式来生成验证码 /*** 使用系统默认字符源生成验…

图形验证码最佳攻略2

下面是注册 如果是手机用户注册,需要发送短信验证码 说明: 发送图形验证码是为了拦截发送短信的.但是不拦截"注册帐号" 但是,用户体验很别扭,因为图形验证码很显然是错误的,但是却可以注册成功. 如果点击注册帐号 ,也要校验图形验证码,那就让用户输入两次图形验证码,…

小程序图形验证码输入校验例子

前言 本教程是基于 “apifm-wxapi” 模块&#xff0c;教你快速实现小程序开发&#xff0c;所以你可能需要先了解以下知识点&#xff1a; 《创建 HelloWorld 项目》《使用 “apifm-wxapi” 快速开发小程序》《免费注册开通后台&#xff0c;获得专属域名》 功能说明 图形验证码的…

Part1:使用 TensorFlow 和 Keras 的 NeRF计算机图形学和深度学习——计算机图形学世界中相机的工作原理

Part1&#xff1a;使用 TensorFlow 和 Keras 的 NeRF计算机图形学和深度学习 1. 效果图2. 原理2.0 前向成像模型2.1 世界坐标系2.2 相机坐标系2.3 坐标变换2.4 投影转换2.5 数据 3. 源码参考 是否有一种方法可以仅从一个场景多张不同视角的照片中捕获整个3D场景&#xff1f; 有…

IDEA+Mysql+Sqlserver安装步骤_kaic

下载Intellij 开发工具&#xff0c;如果有请检查软件是否过期&#xff0c;如果过期卸载电脑上的Intellij软件。卸载步骤&#xff1a; 打开控制面板&#xff0c;选择卸载程序找到Intellij右键卸载 如果没有请保存文档中的Intellij.zip压缩包,进行安装&#xff0c;安装步骤 双…

电脑重置网络

1、键盘WinR键&#xff0c;弹出窗口 2、然后在里面输入cmd输入CMD 3、出现的命令提示框内输入“netsh winsock reset"按”Enter“键 4、重启电脑后生效。

重置电脑

步骤1&#xff1a; 以Win10系统为例 直接在搜索框输入:reset 或者直接选择「设置」-「更新和安全」-「恢复」 步骤2&#xff1a; 步骤3&#xff1a; 根据个人情况选择【保留我的文件】或者【删除所有内容】

如何让Win10 重置此电脑功能无法使用

环境&#xff1a; Win10 专业版 问题描述&#xff1a; 通过设置-更新和安全-恢复-重置此电脑&#xff0c;如何让Win10 重置此电脑功能无法使用 解决方案&#xff1a; 1.PE下删除Win Re 恢复分区&#xff0c;然后重置就无法使用

重置计算机后无法开机,win10重置此电脑失败怎么办_win10重置此电脑失败无法开机修复方法...

当win10系统使用时间长了&#xff0c;难免会有一些电脑故障的出现&#xff0c;这时有些用户就会选择使用重置电脑的方式来解决&#xff0c;但是最近有用户再给自己的win10系统进行重置时总是出现失败的情况&#xff0c;那么win10重置此电脑失败怎么办呢&#xff1f;下面就来告诉…

win10怎么重置计算机,如何重置Win10系统电脑

当系统出现一些难以解决的问题时&#xff0c;我们一般会选择重置电脑来解决问题&#xff0c;比起重装电脑来说简单很多&#xff0c;但是也要注意有可能会失败。下面小编就以win10为例&#xff0c;给大家讲讲怎么重置电脑的操作步骤吧。 操作步骤&#xff1a; 1、打开屏幕侧方的…

DELL 笔记本 - Windows 10 恢复 / 重置此电脑

DELL 笔记本 - Windows 10 恢复 / 重置此电脑 1. Windows 设置 2. 更新和安全 3. 恢复 4. 删除所有内容 5. 所有驱动器 6. 仅删除我的文件 7. 重置 References https://yongqiang.blog.csdn.net/

win10系统如何重置电脑

在win10系统出现系统问题无法修复的时候&#xff0c;我们可以尝试使用win10系统自带的重置功能&#xff0c;相比重装系统在操作上可能要简单不少&#xff0c;不过同样也有失败的风险。下面就让小编带领大家进行一下win10系统重置电脑的操作吧。 win10系统如何重置电脑 1、打…

win10怎样重置电脑

win10怎样重置电脑 windows10提供重置电脑的功能&#xff0c;如果系统遇到问题&#xff0c;运行不正常&#xff0c;或者想刷新系统&#xff0c;可以考虑重置电脑&#xff0c;使电脑有一个全新的干净的系统。下面介绍重置的方法 点击开始菜单&#xff0c;然后选择“设置” 在设…