Java编码技巧:验证码

目录

    • 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目录下的CaptchaConfigKaptchaTextCreator类,以及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:

作用:为上面验证码配置类CaptchaConfiggetKaptchaBeanMath()提供文本实现类

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());}
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DE0开发板交通灯十字路口红绿灯VHDL

名称&#xff1a;基于DE0开发板的交通灯十字路口红绿灯 软件&#xff1a;Quartus 语言&#xff1a;VHDL 要求&#xff1a; 设计一个十字路口交通信号灯的控制电路。分为两种情况&#xff0c;正常状态和报警状态。 1.正常状态&#xff1a;要求红、绿灯按一定的规律亮和灭&a…

触觉智能 PurPle Pi OH(OpenHarmony)开发板

资料汇总 内容预览 产品介绍 PurPle-Pi OH 规格书​​​​​​ 系统编译 Purple-Pi-OH Linux SDK编译 Purple-Pi-OH OHOS SDK编译 使用手册 Purple-Pi-OH Ubuntu系统使用手册 常见FAQ 常见问题 官网 官网地址 Purple Pi OH介绍 Purple Pi OH作为一款兼容树莓派的开…

多个pdf合并成一个文件,3个方法合并pdf

如何把多个pdf合并成一个文件&#xff1f;在我们日常的工作中&#xff0c;经常会遇到一些需要处理的文件&#xff0c;其中包括PDF文件。特别是当我们需要将多个PDF文件合并成一个PDF文件时&#xff0c;会面临一些困难。这样的情况下&#xff0c;我们的阅读能力会受到限制&#…

软件测试中的测试工具和自动化测试

1. 测试工具 测试工具也分为不同人员使用的 开发人员&#xff1a;测试框架&#xff0c;编写测试用例&#xff1b;各类线上dump分析工具如windgb&#xff1b;开发时的集成IDE工具如Visual Studio&#xff0c;idea等等 面向不同测试需求的测试工具 软件测试是软件开发生命周期…

手摸手带你 在Windows系统中安装Istio

Istio简介 通过负载均衡、服务间的身份验证、监控等方法&#xff0c;Istio 可以轻松地创建一个已经部署了服务的网络&#xff0c;而服务的代码只需很少更改甚至无需更改。 通过在整个环境中部署一个特殊的 sidecar 代理为服务添加 Istio 的支持&#xff0c;而代理会拦截微服务…

Multisim14.0仿真(二十七)基于UC3842的反激式开关电源的设计及仿真

一、UC3842简介&#xff1a; UC3842为固定频率电流模式PWM控制器。它们是专门为OFF−线和直流到直流转换器应用与最小的外部组件。内部实现的电路包括用于精确占空比控制的修剪振荡器、温度补偿参考、高增益误差放大器、电流传感比较器和理想适合于驱动功率MOSFET的高电流温度极…

MySQL学习笔记24

MySQL的物理备份&#xff1a; xtrabackup备份介绍&#xff1a; xtrabackup优缺点&#xff1a; 优点&#xff1a; 1、备份过程快速、可靠&#xff08;因为是物理备份&#xff09;&#xff1b;直接拷贝物理文件。 2、支持增量备份&#xff0c;更为灵活&#xff1b; 3、备份…

第十四届蓝桥杯大赛软件赛决赛 C/C++ 大学 B 组 试题 C: 班级活动

[蓝桥杯 2023 国 B] 班级活动 【问题描述】 小明的老师准备组织一次班级活动。班上一共有 n n n 名&#xff08; n n n 为偶数&#xff09;同学&#xff0c;老师想把所有的同学进行分组&#xff0c;每两名同学一组。为了公平&#xff0c;老师给每名同学随机分配了一个 n n …

Eclipse环境基于HDFS的API进行开发

文章目录 IOUtils方式读取文件1.文件准备2.下载安装Eclipse3.打开eclipse&#xff0c;新建java项目&#xff0c;添加关于hadoop的一些包4.包内新建类进行开发5.利用打包的方式生成java jar包6.验证代码正确性 其它问题&#xff1a;Exception in thread “main“ java.lang.Unsu…

【前端】ECMAScript6从入门到进阶

【前端】ECMAScript6从入门到进阶 1.ES6简介及环境搭建 1.1.ECMAScript 6简介 &#xff08;1&#xff09;ECMAScript 6是什么 ECMAScript 6.0&#xff08;以下简称 ES6&#xff09;是 JavaScript 语言的下一代标准&#xff0c;已经在2015年6月正式发布了。它的目标&#xff…

基于微信小程序的刷题考试系统设计与实现(适用于各类考试类、答题类程序)

文章目录 前言系统主要功能&#xff1a;具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计…

【IDEA】使用idea调试时查看对象集合的值

1、在实体类上添加toString方法 2、在要查看集合的地方右键View as→toString 3、View Text复制对象集合的值 4、复制map集合的值同理

[杂谈]-ESP32中的无线通信协议

ESP32中的无线通信协议 文章目录 ESP32中的无线通信协议1、ESP32 无线通信协议简介2、Bluetooth Low Energy (BLE)3、**Bluetooth Classic**4、**ESP-NOW**5、Wi-Fi&#xff08;客户端-服务器通信协议&#xff09;6、MQTT7、**LoRa**8、**GSM/GPRS/LTE**9、总结 ESP32是一个基于…

如何使用ArcGIS Pro直接获取道路中心线

以前使用ArcGIS获取道路中心线&#xff0c;需要先将面要素转换为栅格再获取中心线&#xff0c;现在我们可以通过ArcGIS Pro直接获取道路中心线&#xff0c;这里为大家介绍一下获取方法&#xff0c;希望能对你有所帮助。 新建地理数据库 在存储数据的文件夹上点击右键&#xff…

专栏更新情况:华为流程、产品经理、战略管理、IPD

目录 前言 01 华为流程体系入门课 CSDN学院 02 产品经理进阶课 CSDN学院 03 BLM 战略方法论进阶课 04 IPD 进阶 100 例专栏 作者简介 前言 已上线四大课程专栏更新情况&#xff1a; 01 华为流程体系入门课&#xff08;视频图文&#xff09;&#xff1b; 02 硬件产品经…

Linux文件查找,别名,用户组综合练习

1.文件查看: 查看/etc/passwd文件的第5行 [rootserver ~]# head -5 /etc/passwd root:x:0:0:root:/root:/bin/bash bin:x:1:1:bin:/bin:/sbin/nologin daemon:x:2:2:daemon:/sbin:/sbin/nologin adm:x:3:4:adm:/var/adm:/sbin/nologin lp:x:4:7:lp:/var/spool/lpd:/sbin/nologi…

使用YOLOv5的backbone网络识别图像天气 - P9

目录 环境步骤环境设置包引用声明一个全局的设备 数据准备收集数据集信息构建数据集在数据集中读取分类名称划分训练、测试数据集数据集划分批次 模型设计编写维持卷积前后图像大小不变的padding计算函数编写YOLOv5中使用的卷积模块编写YOLOv5中使用的Bottleneck模块编写YOLOv5…

信息安全:网络物理隔离技术原理与应用.

信息安全&#xff1a;网络物理隔离技术原理与应用. 随着网络攻击技术不断增强&#xff0c;恶意入侵内部网络的风险性也相应急剧提高。满足内外网信息及数据交换需求&#xff0c;又能防止网络安全事件出现的安全技术就应运而生了&#xff0c;这种技术称为“物理隔离技术” 基本原…

企业工程项目管理系统源码(三控:进度组织、质量安全、预算资金成本、二平台:招采、设计管理)

工程项目管理软件&#xff08;工程项目管理系统&#xff09;对建设工程项目管理组织建设、项目策划决策、规划设计、施工建设到竣工交付、总结评估、运维运营&#xff0c;全过程、全方位的对项目进行综合管理 工程项目各模块及其功能点清单 一、系统管理 1、数据字典&am…

照片后期处理软件DxO FilmPack 6 mac中文说明

DxO FilmPack 6 for Mac是一款照片后期处理软件。它可以模拟超过60种著名胶片品牌和类型的色彩和颗粒感&#xff0c;使照片具有复古、艺术和时尚风格。 ​DxO FilmPack 6 mac支持RAW和JPG格式的照片&#xff0c;并提供丰富的调整选项&#xff0c;如亮度、对比度、曝光、阴影和高…