目录
- 1.1、EasyCaptcha(优选,支持种类多,样式多,使用简单)
- 1.1.1、作用
- 1.1.2、官方信息
- 1.1.3、使用案例
- 1.1.4、依赖
- 1.1.5、代码
- 1.1.6、效果
- 1.1.7、拓展
- 1.2、kaptcha
- 1.2.1、作用
- 1.2.2、官方信息
- 1.2.3、使用案例
- 1.2.4、依赖
- 1.2.5、代码
- 1.2.6、效果
- 1.3、AJ-Captcha(TODO)
- 1.3.1、作用
- 1.3.2、官方信息
- 1.3.3、依赖
- 1.3.4、代码
- 1.3.5、效果
- 1.4、tianai-captcha(TODO)
- 1.4.1、作用
- 1.4.2、官方信息
- 1.4.3、依赖
- 1.4.4、代码
- 1.4.5、效果
- 1.5、hutool验证码
- 1.5.1、作用
- 1.5.2、官方信息
- 1.5.3、使用案例
- 1.5.4、依赖
- 1.5.5、代码
- 1.5.6、效果
- 1.5.7、拓展
- 1.6、实战
- 1.6.1、使用场景
- 1.6.2、用途讲解
- 1.6.2.1、流程串讲
- 1.6.2.2、后端验证码校验—网关层校验
- 1.6.2.3、后端验证码校验—直接校验
1.1、EasyCaptcha(优选,支持种类多,样式多,使用简单)
1.1.1、作用
用作Java图形验证码,可用于Java Web、JavaSE等项目,支持情况如下:
- 支持 数字 / 字母 png、数字 / 字母 gif、中文png、中文gif、算术png类型
- 支持设置字段长度、算数数字长度;当然每一种类型都有默认值
- 支持设置字体;当然也有默认字体
- 支持为数字 / 字母 类验证码设置“验证码文本类型”,包含类型有:字母数字混合(默认值)、纯数字、纯字母、纯大写字母、纯小写字母、数字大写字母
效果演示:
1.1.2、官方信息
- gitee项目代码
1.1.3、使用案例
- renren-security
1.1.4、依赖
<dependency>
<groupId>com.github.whvcse</groupId>
<artifactId>easy-captcha</artifactId>
<version>1.6.2</version>
</dependency>
1.1.5、代码
import com.wf.captcha.*;
public class Test {
public static void main(String[] args) {
/*
* 说明:下面一共展示了5种样式的详细使用方法,它们分别是:
* 1、字母/数字 png类型
* 2、字母/数字 gif类型
* 3、中文 png类型
* 4、中文 gif类型
* 5、算术 png类型
*/
// 1、字母/数字 png类型(支持样式:字母数字混合(默认值)、纯数字、纯字母、纯大写字母、纯小写字母、数字大写字母)
SpecCaptcha captcha = new SpecCaptcha();
// 设置验证码显示宽度;宽度默认值:130
// captcha.setWidth(150);
// 设置验证码显示高度;高度默认值:48
// captcha.setHeight(40);
// 设置验证码随机字符长度;默认值:5;比如:asdfg就是5个字符
// captcha.setLen(4);
// 设置验证码文本类型;默认值:TYPE_DEFAULT(字母数字混合)
// captcha.setCharType(Captcha.TYPE_NUM_AND_UPPER);
// 设置验证码的字体;默认值:actionj字体、加粗Font.BOLD、字体大小32磅
// captcha.setFont(new Font("Verdana", Font.PLAIN, 32));
// 验证码结果
System.out.println(">>>>>>>>>>>>>>>>>>>> 字母/数字 png类型 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha.text());
System.out.println("验证码Base64编码:" + captcha.toBase64());
System.out.println(">>>>>>>>>>>>>>>>>>>> 字母/数字 png类型 end <<<<<<<<<<<<<<<<<<<<");
// 2、字母/数字 gif类型(支持样式:字母数字混合(默认值)、纯数字、纯字母、纯大写字母、纯小写字母、数字大写字母)
GifCaptcha captcha2 = new GifCaptcha();
// 设置验证码显示宽度;宽度默认值:130
// captcha2.setWidth(150);
// 设置验证码显示高度;高度默认值:48
// captcha2.setHeight(40);
// 设置验证码随机字符长度;默认值:5;比如:asdfg就是5个字符
// captcha2.setLen(4);
// 设置验证码文本类型;默认值:TYPE_DEFAULT(字母数字混合)
// captcha2.setCharType(Captcha.TYPE_NUM_AND_UPPER);
// 设置验证码的字体;默认值:actionj字体、Font.BOLD(加粗)、32磅
// captcha2.setFont(new Font("Verdana", Font.PLAIN, 32));
// 验证码结果
System.out.println("\n\n\n>>>>>>>>>>>>>>>>>>>> 字母/数字 gif类型 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha2.text());
System.out.println("验证码Base64编码:" + captcha2.toBase64());
System.out.println(">>>>>>>>>>>>>>>>>>>> 字母/数字 gif类型 end <<<<<<<<<<<<<<<<<<<<");
// 3、中文 png类型
ChineseCaptcha captcha3 = new ChineseCaptcha();
// 设置验证码显示宽度;宽度默认值:130
// captcha3.setWidth(150);
// 设置验证码显示高度;高度默认值:48
// captcha3.setHeight(40);
// 设置验证码随机字符长度;默认值:4,详情可看ChineseCaptchaAbstract的无参构造函数;比如:高山流水就是4个字符
// captcha3.setLen(4);
// 设置验证码的字体;默认值:楷体、 Font.PLAIN(不加粗)、 28磅
// captcha3.setFont(new Font("黑体", Font.PLAIN, 32));
// 验证码结果
System.out.println("\n\n\n>>>>>>>>>>>>>>>>>>>> 中文 png类型 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha3.text());
System.out.println("验证码Base64编码:" + captcha3.toBase64());
System.out.println(">>>>>>>>>>>>>>>>>>>> 中文 png类型 end <<<<<<<<<<<<<<<<<<<<");
// 4、中文 gif类型
ChineseGifCaptcha captcha4 = new ChineseGifCaptcha();
// 设置验证码显示宽度;宽度默认值:130
// captcha4.setWidth(150);
// 设置验证码显示高度;高度默认值:48
// captcha4.setHeight(40);
// 设置验证码随机字符长度;默认值:4,详情可看ChineseCaptchaAbstract的无参构造函数;比如:高山流水就是4个字符
// captcha4.setLen(4);
// 设置验证码的字体;默认值:楷体、Font.PLAIN(不加粗)、28磅,详情可看ChineseCaptchaAbstract的无参构造函数
// captcha4.setFont(new Font("黑体", Font.PLAIN, 32));
// 验证码结果
System.out.println("\n\n\n>>>>>>>>>>>>>>>>>>>> 中文 gif类型 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha4.text());
System.out.println("验证码Base64编码:" + captcha4.toBase64());
System.out.println(">>>>>>>>>>>>>>>>>>>> 中文 gif类型 end <<<<<<<<<<<<<<<<<<<<");
// 5、算术 png类型
ArithmeticCaptcha captcha5 = new ArithmeticCaptcha(150, 40);
// 设置验证码显示宽度;宽度默认值:130
// captcha5.setWidth(150);
// 设置验证码显示高度;高度默认值:48
// captcha5.setHeight(40);
// 设置验证码数字长度;默认值:2,详情可看ArithmeticCaptchaAbstract的无参构造函数;比如:“3 + 2 = ?”就是2个字符
// captcha5.setLen(3);
// 设置验证码的字体;默认值:楷体, Font.PLAIN(平常字体), 28
// captcha5.setFont(new Font("黑体", Font.PLAIN, 32));
// 验证码结果
System.out.println("\n\n\n>>>>>>>>>>>>>>>>>>>> 算术 png类型 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha5.text());
System.out.println("验证码内容:" + captcha5.getArithmeticString());
System.out.println("验证码Base64编码:" + captcha5.toBase64());
System.out.println(">>>>>>>>>>>>>>>>>>>> 算术 png类型 end <<<<<<<<<<<<<<<<<<<<");
}
}
1.1.6、效果
查看验证码图片效果方式:
下面返回了Base64编码结果
,如果大家想查看具体图片效果,可以根据下面html
代码新建一个以html
后缀结尾的文件,然后将img
标签中的src
属性替换成验证码Base64编码结果
即可,如下:
<!DOCTYPE HTML>
<html>
<head>
<title>测试验证码</title>
<meta charset="utf-8">
</head>
<body>
<img src="验证码Base64编码"/>
</body>
代码执行结果:
>>>>>>>>>>>>>>>>>>>> 字母/数字 png类型 start <<<<<<<<<<<<<<<<<<<<
验证码结果:utGBu
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 字母/数字 png类型 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 字母/数字 gif类型 start <<<<<<<<<<<<<<<<<<<<
验证码结果:Z3N6N
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 字母/数字 gif类型 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 中文 png类型 start <<<<<<<<<<<<<<<<<<<<
验证码结果:志报背发
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 中文 png类型 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 中文 gif类型 start <<<<<<<<<<<<<<<<<<<<
验证码结果:清却衣是
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 中文 gif类型 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 算术 png类型 start <<<<<<<<<<<<<<<<<<<<
验证码结果:5
验证码内容:9-4=?
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 算术 png类型 end <<<<<<<<<<<<<<<<<<<<
1.1.7、拓展
背景: 上面解释了如何获取验证码结果
以及验证码Base64编码字符串
,其实我们还可以直接把验证码结果输出到响应流中,请看下面代码
代码:
import com.wf.captcha.SpecCaptcha;
@GetMapping("/previewCaptcha")
public void previewCaptcha(HttpServletResponse response) throws IOException {
try (
ServletOutputStream outputStream = response.getOutputStream();
) {
// 将验证码图片返回到前端
response.setContentType("image/png");
SpecCaptcha captcha = new SpecCaptcha(150, 40);
captcha.setLen(4);
// 将验证码图片返回
captcha.out(outputStream);
// 将验证码结果存储到redis中,具体过程省略
String result = captcha.text();
// 存储到redis中……
} catch (Exception e) {
e.printStackTrace();
}
}
结果:
1.2、kaptcha
1.2.1、作用
用作Java图形验证码,可用于Java Web、JavaSE等项目,支持情况如下:
- 支持数字、字母验证码,默认支持
- 支持自定义验证码内容,比如:算数验证码
相比上面EasyCaptcha
来说,这种方式虽然实现比较复杂,但是留给开发者的可修改空间较大,用户可以根据自己的需要去自定义验证码图片样式以及内容等
1.2.2、官方信息
大型项目技术栈第九讲 kaptcha的使用
1.2.3、使用案例
- ruoyi-cloud
1.2.4、依赖
<dependency>
<groupId>pro.fessional</groupId>
<artifactId>kaptcha</artifactId>
<version>2.3.3</version>
</dependency>
1.2.5、代码
以下示例代码来自于: ruoyi-cloud项目的ruoyi-gateway模块下的config
目录下的CaptchaConfig
、KaptchaTextCreator
类,以及service
目录的impl
目录下的ValidateCodeServiceImpl
类
验证码配置类CaptchaConfig:
作用:定义两种类型的验证码,分别是:默认文本验证码、自定义算术验证码
import com.google.code.kaptcha.impl.DefaultKaptcha;
import com.google.code.kaptcha.util.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.Properties;
import static com.google.code.kaptcha.Constants.*;
/**
* 验证码配置类
*
* @author ruoyi
*/
@Configuration
public class CaptchaConfig
{
// 默认文本验证码
@Bean(name = "captchaProducer")
public DefaultKaptcha getKaptchaBean()
{
DefaultKaptcha defaultKaptcha = new DefaultKaptcha();
Properties properties = new Properties();
// 是否有边框 默认为true 我们可以自己设置yes,no
properties.setProperty(KAPTCHA_BORDER, "yes");
// 验证码文本字符颜色 默认为Color.BLACK
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "black");
// 验证码图片宽度 默认为200
properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");
// 验证码图片高度 默认为50
properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");
// 验证码文本字符大小 默认为40
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "38");
// KAPTCHA_SESSION_KEY
properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCode");
// 随机生成字符的范围,默认值:abcde2345678gfynmnpwx
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_STRING, "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ");
// 验证码文本字符长度 默认为5
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "4");
// 验证码文本字体样式 默认为new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize)
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier");
// 图片样式 水纹com.google.code.kaptcha.impl.WaterRipple 鱼眼com.google.code.kaptcha.impl.FishEyeGimpy 阴影com.google.code.kaptcha.impl.ShadowGimpy
properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy");
Config config = new Config(properties);
defaultKaptcha.setConfig(config);
return defaultKaptcha;
}
// 自定义算术验证码
@Bean(name = "captchaProducerMath")
public DefaultKaptcha getKaptchaBeanMath()
{
DefaultKaptcha defaultKaptcha = new DefaultKaptcha();
Properties properties = new Properties();
// 是否有边框 默认为true 我们可以自己设置yes,no
properties.setProperty(KAPTCHA_BORDER, "yes");
// 边框颜色 默认为Color.BLACK
properties.setProperty(KAPTCHA_BORDER_COLOR, "105,179,90");
// 验证码文本字符颜色 默认为Color.BLACK
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "blue");
// 验证码图片宽度 默认为200
properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");
// 验证码图片高度 默认为50
properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");
// 验证码文本字符大小 默认为40
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "35");
// KAPTCHA_SESSION_KEY
properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCodeMath");
// 验证码文本生成器
properties.setProperty(KAPTCHA_TEXTPRODUCER_IMPL, "com.ruoyi.gateway.config.KaptchaTextCreator");
// 验证码文本字符间距 默认为2
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_SPACE, "3");
// 验证码文本字符长度 默认为5
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "6");
// 验证码文本字体样式 默认为new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize)
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier");
// 验证码噪点颜色 默认为Color.BLACK
properties.setProperty(KAPTCHA_NOISE_COLOR, "white");
// 干扰实现类
properties.setProperty(KAPTCHA_NOISE_IMPL, "com.google.code.kaptcha.impl.NoNoise");
// 图片样式 水纹com.google.code.kaptcha.impl.WaterRipple 鱼眼com.google.code.kaptcha.impl.FishEyeGimpy 阴影com.google.code.kaptcha.impl.ShadowGimpy
properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy");
Config config = new Config(properties);
defaultKaptcha.setConfig(config);
return defaultKaptcha;
}
}
算术验证码生成器类KaptchaTextCreator:
作用:为上面验证码配置类CaptchaConfig
中getKaptchaBeanMath()
提供文本实现类
import java.util.Random;
import com.google.code.kaptcha.text.impl.DefaultTextCreator;
/**
* 验证码文本生成器
*
* @author ruoyi
*/
public class KaptchaTextCreator extends DefaultTextCreator
{
private static final String[] CNUMBERS = "0,1,2,3,4,5,6,7,8,9,10".split(",");
@Override
public String getText()
{
Integer result = 0;
Random random = new Random();
int x = random.nextInt(10);
int y = random.nextInt(10);
StringBuilder suChinese = new StringBuilder();
int randomoperands = random.nextInt(3);
if (randomoperands == 0)
{
result = x * y;
suChinese.append(CNUMBERS[x]);
suChinese.append("*");
suChinese.append(CNUMBERS[y]);
}
else if (randomoperands == 1)
{
if ((x != 0) && y % x == 0)
{
result = y / x;
suChinese.append(CNUMBERS[y]);
suChinese.append("/");
suChinese.append(CNUMBERS[x]);
}
else
{
result = x + y;
suChinese.append(CNUMBERS[x]);
suChinese.append("+");
suChinese.append(CNUMBERS[y]);
}
}
else if (randomoperands == 2)
{
if (x >= y)
{
result = x - y;
suChinese.append(CNUMBERS[x]);
suChinese.append("-");
suChinese.append(CNUMBERS[y]);
}
else
{
result = y - x;
suChinese.append(CNUMBERS[y]);
suChinese.append("-");
suChinese.append(CNUMBERS[x]);
}
}
else
{
result = x + y;
suChinese.append(CNUMBERS[x]);
suChinese.append("+");
suChinese.append(CNUMBERS[y]);
}
suChinese.append("=?@" + result);
return suChinese.toString();
}
}
验证码实现处理类ValidateCodeServiceImpl:
说明:
- 针对生成验证码createCaptcha方法来说:首先通过
CaptchaProperties
配置类来判断是否需要生成验证码,以及生成的验证码类型,然后将创建的验证码对应uuid + Base64编码字符串返回给前端,另外一方面把验证码对应uuid+验证码结果存入redis - 针对校验验证码checkCaptcha方法来说:根据用户提交的uuid和验证码结果作为依托,通过uuid去redis中查询真实验证码结果,然后和用户提交的验证码结果作对比,一致说明ok,否则说明用户输入验证码结果有问题,那就需要刷新验证码图片进行重新输入
下面代码中用到的一些内容没有粘贴过来,大家可以去看ruoyi-cloud项目,ValidateCodeServiceImpl类的具体位置是:ValidateCodeServiceImpl
import com.google.code.kaptcha.Producer;
import com.ruoyi.common.core.constant.CacheConstants;
import com.ruoyi.common.core.constant.Constants;
import com.ruoyi.common.core.exception.CaptchaException;
import com.ruoyi.common.core.utils.StringUtils;
import com.ruoyi.common.core.utils.sign.Base64;
import com.ruoyi.common.core.utils.uuid.IdUtils;
import com.ruoyi.common.core.web.domain.AjaxResult;
import com.ruoyi.common.redis.service.RedisService;
import com.ruoyi.gateway.config.properties.CaptchaProperties;
import com.ruoyi.gateway.service.ValidateCodeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.FastByteArrayOutputStream;
import javax.annotation.Resource;
import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.concurrent.TimeUnit;
/**
* 验证码实现处理
*
* @author ruoyi
*/
@Service
public class ValidateCodeServiceImpl implements ValidateCodeService
{
// 数字/字母文本验证码
@Resource(name = "captchaProducer")
private Producer captchaProducer;
// 算术验证码
@Resource(name = "captchaProducerMath")
private Producer captchaProducerMath;
@Autowired
private RedisService redisService;
@Autowired
private CaptchaProperties captchaProperties;
/**
* 生成验证码
*/
@Override
public AjaxResult createCaptcha() throws IOException, CaptchaException
{
AjaxResult ajax = AjaxResult.success();
boolean captchaEnabled = captchaProperties.getEnabled();
ajax.put("captchaEnabled", captchaEnabled);
if (!captchaEnabled)
{
return ajax;
}
// 保存验证码信息
String uuid = IdUtils.simpleUUID();
String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid;
String capStr = null, code = null;
BufferedImage image = null;
String captchaType = captchaProperties.getType();
// 生成验证码
if ("math".equals(captchaType))
{
String capText = captchaProducerMath.createText();
capStr = capText.substring(0, capText.lastIndexOf("@"));
code = capText.substring(capText.lastIndexOf("@") + 1);
image = captchaProducerMath.createImage(capStr);
}
else if ("char".equals(captchaType))
{
capStr = code = captchaProducer.createText();
image = captchaProducer.createImage(capStr);
}
redisService.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES);
// 转换流信息写出
FastByteArrayOutputStream os = new FastByteArrayOutputStream();
try
{
ImageIO.write(image, "jpg", os);
}
catch (IOException e)
{
return AjaxResult.error(e.getMessage());
}
ajax.put("uuid", uuid);
ajax.put("img", Base64.encode(os.toByteArray()));
return ajax;
}
/**
* 校验验证码
*/
@Override
public void checkCaptcha(String code, String uuid) throws CaptchaException
{
if (StringUtils.isEmpty(code))
{
throw new CaptchaException("验证码不能为空");
}
if (StringUtils.isEmpty(uuid))
{
throw new CaptchaException("验证码已失效");
}
String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid;
String captcha = redisService.getCacheObject(verifyKey);
redisService.deleteObject(verifyKey);
if (!code.equalsIgnoreCase(captcha))
{
throw new CaptchaException("验证码错误");
}
}
}
1.2.6、效果
算数验证码:
文本验证码:
1.3、AJ-Captcha(TODO)
1.3.1、作用
支持滑动拼图
和文字点选
这两种方式验证码类型
1.3.2、官方信息
- gitee代码
1.3.3、依赖
整理麻烦,以后在整理
1.3.4、代码
整理麻烦,以后在整理
1.3.5、效果
整理麻烦,以后在整理
1.4、tianai-captcha(TODO)
1.4.1、作用
支持多种验证码方式,包括:
- 滑块验证码
- 旋转验证码
- 滑动还原验证码
- 文字点选验证码
实现效果如下图:
1.4.2、官方信息
- gitee项目
1.4.3、依赖
整理麻烦,以后在整理
1.4.4、代码
整理麻烦,以后在整理
1.4.5、效果
整理麻烦,以后在整理
1.5、hutool验证码
1.5.1、作用
用作Java图形验证码,可用于Java Web、JavaSE等项目,支持情况如下:
- 支持验证码图片的线段干扰、圆圈干扰、扭曲干扰3种干扰方式
- 默认情况下生成数字 / 字母混合的验证码图片,支持自定义验证码内容
1.5.2、官方信息
- hutool图形验证码文档
1.5.3、使用案例
- Snowy-Cloud
1.5.4、依赖
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.16</version>
</dependency>
1.5.5、代码
import cn.hutool.captcha.CaptchaUtil;
import cn.hutool.captcha.CircleCaptcha;
import cn.hutool.captcha.LineCaptcha;
import cn.hutool.captcha.ShearCaptcha;
import cn.hutool.captcha.generator.RandomGenerator;
public class Test {
public static void main(String[] args) {
// 线段干扰验证码
// 参数含义:图片宽度、图片高度、字符个数、干扰线条数
LineCaptcha captcha = CaptchaUtil.createLineCaptcha(100, 38, 4, 4);
// 验证码结果
System.out.println(">>>>>>>>>>>>>>>>>>>> 线段干扰验证码 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha.getCode());
System.out.println("验证码Base64编码:" + captcha.getImageBase64Data());
System.out.println(">>>>>>>>>>>>>>>>>>>> 线段干扰验证码 end <<<<<<<<<<<<<<<<<<<<\n\n\n");
// 圆圈干扰验证码
// 参数含义:图片宽度、图片高度、字符个数、干扰圆圈条数
CircleCaptcha captcha2 = CaptchaUtil.createCircleCaptcha(100, 38, 4, 5);
// 验证码结果
System.out.println(">>>>>>>>>>>>>>>>>>>> 圆圈干扰验证码 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha2.getCode());
System.out.println("验证码Base64编码:" + captcha2.getImageBase64Data());
System.out.println(">>>>>>>>>>>>>>>>>>>> 圆圈干扰验证码 end <<<<<<<<<<<<<<<<<<<<\n\n\n");
// 扭曲干扰验证码
// 参数含义:图片宽度、图片高度、字符个数、干扰线宽度
ShearCaptcha captcha3 = CaptchaUtil.createShearCaptcha(100, 38, 4, 5);
// 验证码结果
System.out.println(">>>>>>>>>>>>>>>>>>>> 扭曲干扰验证码 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha3.getCode());
System.out.println("验证码Base64编码:" + captcha3.getImageBase64Data());
System.out.println(">>>>>>>>>>>>>>>>>>>> 扭曲干扰验证码 end <<<<<<<<<<<<<<<<<<<<\n\n\n");
// 说明:自定义验证码代码以线段干扰验证码LineCaptcha为例,讲解相关使用方法,其中LineCaptcha可以替换成另外两种验证码类型,这里不在演示
// 自定义纯数字验证码(hutool官方)
// 参数含义:图片宽度、图片高度、字符个数、干扰线条数
LineCaptcha captcha4 = CaptchaUtil.createLineCaptcha(100, 38, 4, 4);
// 自定义纯数字的验证码(随机4位数字,可重复)
captcha4.setGenerator(new RandomGenerator("0123456789", 4));
// 验证码结果
System.out.println(">>>>>>>>>>>>>>>>>>>> 自定义纯数字验证码 start <<<<<<<<<<<<<<<<<<<<");
System.out.println("验证码结果:" + captcha4.getCode());
System.out.println("验证码Base64编码:" + captcha4.getImageBase64Data());
System.out.println(">>>>>>>>>>>>>>>>>>>> 自定义纯数字验证码 end <<<<<<<<<<<<<<<<<<<<");
}
}
1.5.6、效果
下面返回了Base64编码结果
,如果大家想查看具体图片效果,可以根据下面html
代码新建一个以html
后缀结尾的文件,然后将img
标签中的src
属性替换成验证码Base64编码结果
即可,如下:
<!DOCTYPE HTML>
<html>
<head>
<title>测试验证码</title>
<meta charset="utf-8">
</head>
<body>
<img src="验证码Base64编码"/>
</body>
代码执行结果:
>>>>>>>>>>>>>>>>>>>> 线段干扰验证码 start <<<<<<<<<<<<<<<<<<<<
验证码结果:4ose
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 线段干扰验证码 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 圆圈干扰验证码 start <<<<<<<<<<<<<<<<<<<<
验证码结果:s024
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 圆圈干扰验证码 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 扭曲干扰验证码 start <<<<<<<<<<<<<<<<<<<<
验证码结果:ry5n
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 扭曲干扰验证码 end <<<<<<<<<<<<<<<<<<<<
>>>>>>>>>>>>>>>>>>>> 自定义纯数字验证码 start <<<<<<<<<<<<<<<<<<<<
验证码结果:7231
验证码Base64编码:
>>>>>>>>>>>>>>>>>>>> 自定义纯数字验证码 end <<<<<<<<<<<<<<<<<<<<
1.5.7、拓展
背景: 上面解释了如何获取验证码结果
以及验证码Base64编码字符串
,其实我们还可以直接把验证码结果输出到响应流中,请看下面代码
代码:
import cn.hutool.captcha.CaptchaUtil;
import cn.hutool.captcha.LineCaptcha;
@GetMapping("/previewCaptcha")
public void previewCaptcha(HttpServletResponse response) throws IOException {
try (
ServletOutputStream outputStream = response.getOutputStream();
) {
// 将验证码图片返回到前端
response.setContentType("image/png");
LineCaptcha captcha = CaptchaUtil.createLineCaptcha(100, 38, 4, 4);
// 将验证码图片返回
captcha.write(outputStream);
// 将验证码结果存储到redis中,具体过程省略
String result = captcha.getCode();
// 存储到redis中……
} catch (Exception e) {
e.printStackTrace();
}
}
结果:
1.6、实战
1.6.1、使用场景
- 用户登录
- 注册账户
1.6.2、用途讲解
1.6.2.1、流程串讲
大家都看过验证码页面,比如ruoyi的登录页面:
所以需要用户进入该页面时就能把验证码给展示出来,这涉及到验证码图片获取接口,我已经在上面讲述了如何生成验证码图片的base64编码
考虑到用户需要提交用户输入验证码结果进行比对,我们在生成验证码的时候需要把验证码结果存储起来,这一般使用两种方式
- 存储到session中:由于在分布式微服务应用中很少使用共享session方式了,所以这种一般不再使用
- 存储到redis中:在存储的时候,我们可以把uuid和验证码结果进行绑定存储到redis中,当把验证码结果返回给用户的时候,同时把和验证码结果返回的uuid返回给用户
下面以把数据存储到redis中的方式进行讲解后续的步骤
以登录场景为例,用户输入用户名、密码、验证码之后点击登录按钮,由于我们之前已经把和验证码一一对应的uuid返回给用户,所以前端同事在提交数据的时候需要也需要把uuid提交到后端,进而后端依次完成验证码、登录信息的校验
这时候流程走到了后端代码中,我们本次讲解验证码的验证过程,可以分为这么几种情况
- 网关层校验:适合分布式微服务代码,在gateway网关层完成验证码的校验工作,比如:ruoyi-cloud
- 过滤器校验:都挺适合,可以把过滤器代码放在网关模块中完成校验工作
- 直接校验:都挺适合,但是不够优雅,比如:renren-security
1.6.2.2、后端验证码校验—网关层校验
本次以ruoyi-cloud为例讲解
网关层拦截获取验证码请求:
作用:在网关层完成验证码获取工作
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.server.RequestPredicates;
import org.springframework.web.reactive.function.server.RouterFunction;
import org.springframework.web.reactive.function.server.RouterFunctions;
import com.ruoyi.gateway.handler.ValidateCodeHandler;
/**
* 路由配置信息
*
* @author ruoyi
*/
@Configuration
public class RouterFunctionConfiguration
{
@Autowired
private ValidateCodeHandler validateCodeHandler;
@SuppressWarnings("rawtypes")
@Bean
public RouterFunction routerFunction()
{
return RouterFunctions.route(
RequestPredicates.GET("/code").and(RequestPredicates.accept(MediaType.TEXT_PLAIN)),
validateCodeHandler);
}
}
处理获取验证码请求:
作用:获取验证码,一方面将验证码和对应uuid返回给前端,另外一方面把uuid和验证码进行绑定并存储在redis中
import java.io.IOException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.reactive.function.BodyInserters;
import org.springframework.web.reactive.function.server.HandlerFunction;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.reactive.function.server.ServerResponse;
import com.ruoyi.common.core.exception.CaptchaException;
import com.ruoyi.common.core.web.domain.AjaxResult;
import com.ruoyi.gateway.service.ValidateCodeService;
import reactor.core.publisher.Mono;
/**
* 验证码获取
*
* @author ruoyi
*/
@Component
public class ValidateCodeHandler implements HandlerFunction<ServerResponse>
{
@Autowired
private ValidateCodeService validateCodeService;
@Override
public Mono<ServerResponse> handle(ServerRequest serverRequest)
{
AjaxResult ajax;
try
{
ajax = validateCodeService.createCaptcha();
}
catch (CaptchaException | IOException e)
{
return Mono.error(e);
}
return ServerResponse.status(HttpStatus.OK).body(BodyInserters.fromValue(ajax));
}
}
生成验证码和校验验证码接口:
作用:为上面提供生成验证码服务、当用户提交登录、注册请求时,用于校验验证码是否正确
import java.io.IOException;
import com.ruoyi.common.core.exception.CaptchaException;
import com.ruoyi.common.core.web.domain.AjaxResult;
/**
* 验证码处理
*
* @author ruoyi
*/
public interface ValidateCodeService
{
/**
* 生成验证码
*/
public AjaxResult createCaptcha() throws IOException, CaptchaException;
/**
* 校验验证码
*/
public void checkCaptcha(String key, String value) throws CaptchaException;
}
生成验证码和校验验证码实现类:
import com.google.code.kaptcha.Producer;
import com.ruoyi.common.core.constant.CacheConstants;
import com.ruoyi.common.core.constant.Constants;
import com.ruoyi.common.core.exception.CaptchaException;
import com.ruoyi.common.core.utils.StringUtils;
import com.ruoyi.common.core.utils.sign.Base64;
import com.ruoyi.common.core.utils.uuid.IdUtils;
import com.ruoyi.common.core.web.domain.AjaxResult;
import com.ruoyi.common.redis.service.RedisService;
import com.ruoyi.gateway.config.properties.CaptchaProperties;
import com.ruoyi.gateway.service.ValidateCodeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.FastByteArrayOutputStream;
import javax.annotation.Resource;
import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.concurrent.TimeUnit;
/**
* 验证码实现处理
*
* @author ruoyi
*/
@Service
public class ValidateCodeServiceImpl implements ValidateCodeService
{
@Resource(name = "captchaProducer")
private Producer captchaProducer;
@Resource(name = "captchaProducerMath")
private Producer captchaProducerMath;
@Autowired
private RedisService redisService;
@Autowired
private CaptchaProperties captchaProperties;
/**
* 生成验证码
*/
@Override
public AjaxResult createCaptcha() throws IOException, CaptchaException
{
AjaxResult ajax = AjaxResult.success();
boolean captchaEnabled = captchaProperties.getEnabled();
ajax.put("captchaEnabled", captchaEnabled);
if (!captchaEnabled)
{
return ajax;
}
// 保存验证码信息
String uuid = IdUtils.simpleUUID();
String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid;
String capStr = null, code = null;
BufferedImage image = null;
String captchaType = captchaProperties.getType();
// 生成验证码
if ("math".equals(captchaType))
{
String capText = captchaProducerMath.createText();
capStr = capText.substring(0, capText.lastIndexOf("@"));
code = capText.substring(capText.lastIndexOf("@") + 1);
image = captchaProducerMath.createImage(capStr);
}
else if ("char".equals(captchaType))
{
capStr = code = captchaProducer.createText();
image = captchaProducer.createImage(capStr);
}
redisService.setCacheObject(verifyKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES);
// 转换流信息写出
FastByteArrayOutputStream os = new FastByteArrayOutputStream();
try
{
ImageIO.write(image, "jpg", os);
}
catch (IOException e)
{
return AjaxResult.error(e.getMessage());
}
ajax.put("uuid", uuid);
ajax.put("img", Base64.encode(os.toByteArray()));
return ajax;
}
/**
* 校验验证码
*/
@Override
public void checkCaptcha(String code, String uuid) throws CaptchaException
{
if (StringUtils.isEmpty(code))
{
throw new CaptchaException("验证码不能为空");
}
if (StringUtils.isEmpty(uuid))
{
throw new CaptchaException("验证码已失效");
}
String verifyKey = CacheConstants.CAPTCHA_CODE_KEY + uuid;
String captcha = redisService.getCacheObject(verifyKey);
redisService.deleteObject(verifyKey);
if (!code.equalsIgnoreCase(captcha))
{
throw new CaptchaException("验证码错误");
}
}
}
拦截登录、注册请求,校验验证码是否正确过滤器:
作用:拦截登录、注册请求,统一校验验证码是否正确
/**
* 验证码过滤器
*
* @author ruoyi
*/
@Component
public class ValidateCodeFilter extends AbstractGatewayFilterFactory<Object>
{
private final static String[] VALIDATE_URL = new String[] { "/auth/login", "/auth/register" };
@Autowired
private ValidateCodeService validateCodeService;
@Autowired
private CaptchaProperties captchaProperties;
private static final String CODE = "code";
private static final String UUID = "uuid";
@Override
public GatewayFilter apply(Object config)
{
return (exchange, chain) -> {
ServerHttpRequest request = exchange.getRequest();
// 非登录/注册请求或验证码关闭,不处理
if (!StringUtils.containsAnyIgnoreCase(request.getURI().getPath(), VALIDATE_URL) || !captchaProperties.getEnabled())
{
return chain.filter(exchange);
}
try
{
String rspStr = resolveBodyFromRequest(request);
JSONObject obj = JSON.parseObject(rspStr);
validateCodeService.checkCaptcha(obj.getString(CODE), obj.getString(UUID));
}
catch (Exception e)
{
return ServletUtils.webFluxResponseWriter(exchange.getResponse(), e.getMessage());
}
return chain.filter(exchange);
};
}
private String resolveBodyFromRequest(ServerHttpRequest serverHttpRequest)
{
// 获取请求体
Flux<DataBuffer> body = serverHttpRequest.getBody();
AtomicReference<String> bodyRef = new AtomicReference<>();
body.subscribe(buffer -> {
CharBuffer charBuffer = StandardCharsets.UTF_8.decode(buffer.asByteBuffer());
DataBufferUtils.release(buffer);
bodyRef.set(charBuffer.toString());
});
return bodyRef.get();
}
}
1.6.2.3、后端验证码校验—直接校验
本次以renren-security为例讲解
获取验证码方法:
import io.renren.common.exception.ErrorCode;
import io.renren.common.validator.AssertUtils;
import io.renren.modules.security.service.CaptchaService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiImplicitParam;
import io.swagger.annotations.ApiOperation;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* 登录
*
* @author Mark sunlightcs@gmail.com
*/
@RestController
@Api(tags="登录管理")
public class LoginController {
@Autowired
private CaptchaService captchaService;
@GetMapping("captcha")
@ApiOperation(value = "验证码", produces="application/octet-stream")
@ApiImplicitParam(paramType = "query", dataType="string", name = "uuid", required = true)
public void captcha(HttpServletResponse response, String uuid)throws IOException {
//uuid不能为空
AssertUtils.isBlank(uuid, ErrorCode.IDENTIFIER_NOT_NULL);
//生成验证码
captchaService.create(response, uuid);
}
}
获取验证码和校验验证码接口:
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* 验证码
*
* @author Mark sunlightcs@gmail.com
*/
public interface CaptchaService {
/**
* 图片验证码
*/
void create(HttpServletResponse response, String uuid) throws IOException;
/**
* 验证码效验
* @param uuid uuid
* @param code 验证码
* @return true:成功 false:失败
*/
boolean validate(String uuid, String code);
}
获取验证码和校验验证码实现类:
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.wf.captcha.ArithmeticCaptcha;
import com.wf.captcha.SpecCaptcha;
import com.wf.captcha.base.Captcha;
import io.renren.common.redis.RedisKeys;
import io.renren.common.redis.RedisUtils;
import io.renren.modules.security.service.CaptchaService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.concurrent.TimeUnit;
/**
* 验证码
*
* @author Mark sunlightcs@gmail.com
*/
@Service
public class CaptchaServiceImpl implements CaptchaService {
@Autowired
private RedisUtils redisUtils;
@Value("${renren.redis.open: false}")
private boolean open;
/**
* Local Cache 5分钟过期
*/
Cache<String, String> localCache = CacheBuilder.newBuilder().maximumSize(1000).expireAfterAccess(5, TimeUnit.MINUTES).build();
@Override
public void create(HttpServletResponse response, String uuid) throws IOException {
response.setContentType("image/gif");
response.setHeader("Pragma", "No-cache");
response.setHeader("Cache-Control", "no-cache");
response.setDateHeader("Expires", 0);
//生成验证码
SpecCaptcha captcha = new SpecCaptcha(150, 40);
captcha.setLen(5);
captcha.setCharType(Captcha.TYPE_DEFAULT);
captcha.out(response.getOutputStream());
//保存到Redis缓存
setCache(uuid, captcha.text());
}
@Override
public boolean validate(String uuid, String code) {
//获取验证码
String captcha = getCache(uuid);
//效验成功
if(code.equalsIgnoreCase(captcha)){
return true;
}
return false;
}
private void setCache(String key, String value){
if(open){
key = RedisKeys.getCaptchaKey(key);
redisUtils.set(key, value, 300);
}else{
localCache.put(key, value);
}
}
private String getCache(String key){
if(open){
key = RedisKeys.getCaptchaKey(key);
String captcha = (String)redisUtils.get(key);
//删除验证码
if(captcha != null){
redisUtils.delete(key);
}
return captcha;
}
String captcha = localCache.getIfPresent(key);
//删除验证码
if(captcha != null){
localCache.invalidate(key);
}
return captcha;
}
}
在登陆接口中校验验证码是否正确:
import io.renren.common.exception.ErrorCode;
import io.renren.common.validator.AssertUtils;
import io.renren.modules.security.service.CaptchaService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiImplicitParam;
import io.swagger.annotations.ApiOperation;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* 登录
*
* @author Mark sunlightcs@gmail.com
*/
@RestController
@Api(tags="登录管理")
public class LoginController {
@Autowired
private CaptchaService captchaService;
@PostMapping("login")
@ApiOperation(value = "登录")
public Result login(HttpServletRequest request, @RequestBody LoginDTO login) {
//效验数据
ValidatorUtils.validateEntity(login);
//验证码是否正确
boolean flag = captchaService.validate(login.getUuid(), login.getCaptcha());
if(!flag){
return new Result().error(ErrorCode.CAPTCHA_ERROR);
}
//用户信息
SysUserDTO user = sysUserService.getByUsername(login.getUsername());
SysLogLoginEntity log = new SysLogLoginEntity();
log.setOperation(LoginOperationEnum.LOGIN.value());
log.setCreateDate(new Date());
log.setIp(IpUtils.getIpAddr(request));
log.setUserAgent(request.getHeader(HttpHeaders.USER_AGENT));
log.setIp(IpUtils.getIpAddr(request));
//用户不存在
if(user == null){
log.setStatus(LoginStatusEnum.FAIL.value());
log.setCreatorName(login.getUsername());
sysLogLoginService.save(log);
throw new RenException(ErrorCode.ACCOUNT_PASSWORD_ERROR);
}
//密码错误
if(!PasswordUtils.matches(login.getPassword(), user.getPassword())){
log.setStatus(LoginStatusEnum.FAIL.value());
log.setCreator(user.getId());
log.setCreatorName(user.getUsername());
sysLogLoginService.save(log);
throw new RenException(ErrorCode.ACCOUNT_PASSWORD_ERROR);
}
//账号停用
if(user.getStatus() == UserStatusEnum.DISABLE.value()){
log.setStatus(LoginStatusEnum.LOCK.value());
log.setCreator(user.getId());
log.setCreatorName(user.getUsername());
sysLogLoginService.save(log);
throw new RenException(ErrorCode.ACCOUNT_DISABLE);
}
//登录成功
log.setStatus(LoginStatusEnum.SUCCESS.value());
log.setCreator(user.getId());
log.setCreatorName(user.getUsername());
sysLogLoginService.save(log);
return sysUserTokenService.createToken(user.getId());
}
}