目录
- 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,noproperties.setProperty(KAPTCHA_BORDER, "yes");// 验证码文本字符颜色 默认为Color.BLACKproperties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "black");// 验证码图片宽度 默认为200properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");// 验证码图片高度 默认为50properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");// 验证码文本字符大小 默认为40properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "38");// KAPTCHA_SESSION_KEYproperties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCode");// 随机生成字符的范围,默认值:abcde2345678gfynmnpwxproperties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_STRING, "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ");// 验证码文本字符长度 默认为5properties.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.ShadowGimpyproperties.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,noproperties.setProperty(KAPTCHA_BORDER, "yes");// 边框颜色 默认为Color.BLACKproperties.setProperty(KAPTCHA_BORDER_COLOR, "105,179,90");// 验证码文本字符颜色 默认为Color.BLACKproperties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "blue");// 验证码图片宽度 默认为200properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");// 验证码图片高度 默认为50properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");// 验证码文本字符大小 默认为40properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "35");// KAPTCHA_SESSION_KEYproperties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCodeMath");// 验证码文本生成器properties.setProperty(KAPTCHA_TEXTPRODUCER_IMPL, "com.ruoyi.gateway.config.KaptchaTextCreator");// 验证码文本字符间距 默认为2properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_SPACE, "3");// 验证码文本字符长度 默认为5properties.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.BLACKproperties.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.ShadowGimpyproperties.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(",");@Overridepublic 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;@Autowiredprivate RedisService redisService;@Autowiredprivate CaptchaProperties captchaProperties;/*** 生成验证码*/@Overridepublic 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;}/*** 校验验证码*/@Overridepublic 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
{@Autowiredprivate ValidateCodeHandler validateCodeHandler;@SuppressWarnings("rawtypes")@Beanpublic 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>
{@Autowiredprivate ValidateCodeService validateCodeService;@Overridepublic 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;@Autowiredprivate RedisService redisService;@Autowiredprivate CaptchaProperties captchaProperties;/*** 生成验证码*/@Overridepublic 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;}/*** 校验验证码*/@Overridepublic 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" };@Autowiredprivate ValidateCodeService validateCodeService;@Autowiredprivate CaptchaProperties captchaProperties;private static final String CODE = "code";private static final String UUID = "uuid";@Overridepublic 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 {@Autowiredprivate 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 {@Autowiredprivate 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();@Overridepublic 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());}@Overridepublic 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 {@Autowiredprivate 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());}
}