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.mzph.cn/news/90690.shtml

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

相关文章

SLAM从入门到精通(amcl定位使用)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 学习slam&#xff0c;一般就是所谓的边定位、边制图的知识。然而在实际生产过程中&#xff0c;比如扫地机器人、agv、巡检机器人、农业机器人&…

聊一聊JDK21-虚拟线程

目录 前言 Virtual Threads的开始 为什么需要Virtual Threads JDK19 预览版初次出现 JDK21 Virtual Threads的正式发布 Virtual Threads 该怎么使用 简单聊聊Virtual Threads的实现 使用时候的注意事项 本地尝鲜一下JDK21及Virtual Threads 结语 前言 2023年9月19日…

字符串改错题(找出代码中所有错误,将一个字符串倒序)

#include "string.h" main() {char *pSrc "hello,world";char *pDest NULL; int iLen strlen(pSrc);pDest (char*)malloc(iLen);char *pD pDest;char* pS pSrc[iLen]; while(iLen--!0){pD pS--;}printf("%s", pDest);return 0; }在使用 m…

C理解(五):编译,链接库,宏,关键字,变量

编译 编译过程 文件.c->(预处理)->文件.i->(编译)->文件.S->(汇编)->文件.o->(链接)->elf程序 预处理 内容:加载头文件(#include),清除注释(//,./*),替换条件编译(#if #elif #endif #ifdef),替换宏定义(#define) …

探索社会工程的深度:从定义到高级攻击策略

在广阔的网络安全领域&#xff0c;社会工程作为一种微妙的威胁而出现&#xff0c;它利用人类的漏洞来访问敏感信息或实施欺诈。网络安全背景下的社会工程的定义很明确&#xff1a;它包括使用欺骗手段操纵个人泄露机密或个人信息&#xff0c;然后将这些信息用于欺诈目的。 此类…

【AI视野·今日CV 计算机视觉论文速览 第257期】Fri, 29 Sep 2023

AI视野今日CS.CV 计算机视觉论文速览 Fri, 29 Sep 2023 Totally 99 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Computer Vision Papers Learning to Transform for Generalizable Instance-wise Invariance Authors Utkarsh Singhal, Carlos Esteves, Ameesh M…

WebGL笔记:绘制多个点,三角形,以及画各种不同的线条

绘制多点 1 &#xff09; WebGL 缓冲区 我们在用js定点位的时候&#xff0c;肯定是要建立一份顶点数据的&#xff0c;这份顶点数据是给着色器的&#xff0c;因为着色器需要这份顶点数据绘图然而&#xff0c;我们在js中建立顶点数据&#xff0c;着色器肯定是拿不到的&#xff…

《Python趣味工具》——ppt的操作(刷题版)

前面我们对PPT进行了一定的操作&#xff0c;并将其中的文字提取到了word文档中。现在就让我们来刷几道题巩固巩固吧&#xff01; 文章目录 1. 查看PPT&#xff08;上&#xff09;2. 查看PPT&#xff08;中&#xff09;3. 查看PPT&#xff08;下&#xff09;4. PPT的页码5. 大学…

安防监控/视频汇聚平台EasyCVR云端录像不展示是什么原因?该如何解决?

视频云存储/安防监控EasyCVR视频汇聚平台基于云边端智能协同&#xff0c;支持海量视频的轻量化接入与汇聚、转码与处理、全网智能分发、视频集中存储等。音视频流媒体视频平台EasyCVR拓展性强&#xff0c;视频能力丰富&#xff0c;具体可实现视频监控直播、视频轮播、视频录像、…

Excel技巧之【锁定工作簿】

Excel工作簿是Excel工作区中一个或多个工作表的集合&#xff0c;我们知道Excel可以设置锁定工作表&#xff0c;防止意外或被他人修改&#xff0c;但可能有小伙伴不知道&#xff0c;Excel工作簿也同样可以设置锁定&#xff0c;防止更改。 那工作簿锁定后会怎么样呢&#xff1f;…

【数据结构】顺序查找,折半查找,分块查找的知识点总结及相应的代码实现

目录 1、顺序查找 定义及步骤 代码实现 2、折半查找 定义及步骤 代码实现 折半查找判定树 3、分块查找 定义及步骤 1、顺序查找 定义及步骤 顺序查找的定义&#xff1a;从数据集合的起始位置开始&#xff0c;逐一比较每个数据元素&#xff0c;直到找到所要查找…

uni-app 实现凸起的 tabbar 底部导航栏

效果图 在 pages.json 中设置隐藏自带的 tabbar 导航栏 "custom": true, // 开启自定义tabBar(不填每次原来的tabbar在重新加载时都回闪现) 新建一个 custom-tabbar.vue 自定义组件页面 custom-tabbar.vue <!-- 自定义底部导航栏 --> <template><v…

react.js在visual code 下的hello World

想学习reacr.js &#xff0c;就开始做一个hello world。 我的环境是visual code &#xff0c;所以我找这个环境下的例子。参照&#xff1a; https://code.visualstudio.com/docs/nodejs/reactjs-tutorial 要学习react.js &#xff0c;还得先安装node.js&#xff0c;我在visual …

github代码提交过程详细介绍

1、下载github上面的代码 &#xff08;1&#xff09;在github网站上&#xff0c;找到想要下载的代码仓库界面&#xff0c;点击Code选项就可以看到仓库的git下载地址&#xff1b; &#xff08;2&#xff09;使用命令下载&#xff1a;git clone 地址&#xff1b; 2、配置本地git…

【通意千问】大模型GitHub开源工程学习笔记(1)

9月25日&#xff0c;阿里云开源通义千问140亿参数模型Qwen-14B及其对话模型Qwen-14B-Chat,免费可商用。 立马就到了GitHub去fork。 GitHub&#xff1a; GitHub - QwenLM/Qwen: The official repo of Qwen (通义千问) chat & pretrained large language model proposed b…

在vue使用wangEditor(简单使用)

wangEditor不同的版本使用方法都不一样&#xff0c;这里以目前最新的参考官网方法使用2023-09-28 首先安装&#xff0c;参考官网&#xff0c;注意editor跟editor-for-vue两个都要装 yarn add wangeditor/editor # 或者 npm install wangeditor/editor --saveyarn add wangedit…

【Linux指令集】---git命令的基本使用

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【Linux专栏】&#x1f388; 本专栏旨在分享学习Linux的一点学习心得&#xff0c;欢迎大家在评论区讨论&#x1f48c; 演示环境&#xff1…

Event Loop——事件循环

JS 是单线程的&#xff0c;也就是同一个时刻只能做一件事情。 那么思考&#xff1a;为什么浏览器可以同时执行异步任务呢&#xff1f;因为浏览器是多线程的。 当 JS 需要执行异步任务时&#xff0c;浏览器会另外启动一个线程去执行该任务。 也就是说&#xff0c;“JS 是单线程…

Springboot中slf4j日志的简单应用

1、注入依赖&#xff08;pom.xml&#xff09; <!-- https://mvnrepository.com/artifact/org.slf4j/slf4j-api --> <dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version>2.0.9</version> &…

【5G PHY】物理层逻辑和物理天线的映射

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…