spring Boot手把手教学(11):生成动态验证码并前后端校验

spring Boot手把手教学(11):生成动态验证码并前后端校验在前端页面中,只要把 URL 放到 image 的 URL 中,即可显示,这里不再演示。

最近需要生成一个动态的验证码,在登录页面使用,并在前后端进行校验;

spring Boot手把手教学(11):生成动态验证码并前后端校验

实现原理:

后端生成动态二维码,存储在 session 里面;

前端调取接口,展示在登录页面;

前端登录时候,把验证码传给后端,后端和 session 里面的值进行对比。

1 生成动态验证码图片

新建一个 classValidateCode:

package hello;

import org.apache.commons.io.FileUtils;

import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.Base64;
import java.util.HashMap;
import java.util.Map;
import java.util.Random;

//  生成随机验证码
public class ValidateCode {
    
    private static Random random = new Random();
    private int width = 160;// 宽
    private int height = 40;// 高
    private int lineSize = 30;// 干扰线数量
    private int stringNum = 4;//随机产生字符的个数
    
    private String randomString = "0123456789abcdefghijklmnopqrstuvwxyz";
    
    private final String sessionKey = "RANDOMKEY";
    
    
    /*
     *  获取字体
     */
    private Font getFont() {
        return new Font("Times New Roman", Font.ROMAN_BASELINE, 40);
    }
    
    /*
     *  获取颜色
     */
    private static Color getRandomColor(int fc, int bc) {
        
        fc = Math.min(fc, 255);
        bc = Math.min(bc, 255);
        
        int r = fc + random.nextInt(bc - fc - 16);
        int g = fc + random.nextInt(bc - fc - 14);
        int b = fc + random.nextInt(bc - fc - 12);
        
        return new Color(r, g, b);
    }
    
    /*
     *  绘制干扰线
     */
    private void drawLine(Graphics g) {
        int x = random.nextInt(width);
        int y = random.nextInt(height);
        int xl = random.nextInt(20);
        int yl = random.nextInt(10);
        g.drawLine(x, y, x + xl, y + yl);
    }
    
    /*
     *  获取随机字符
     */
    private String getRandomString(int num) {
        num = num > 0 ? num : randomString.length();
        return String.valueOf(randomString.charAt(random.nextInt(num)));
    }
    
    /*
     *  绘制字符串
     */
    private String drawString(Graphics g, String randomStr, int i) {
        g.setFont(getFont());
        g.setColor(getRandomColor(108, 190));
        System.out.println(random.nextInt(randomString.length()));
        String rand = getRandomString(random.nextInt(randomString.length()));
        randomStr += rand;
        g.translate(random.nextInt(3), random.nextInt(6));
        g.drawString(rand, 40 * i + 10, 25);
        return randomStr;
    }
    
    /*
     *  生成随机图片
     */
     public void getRandomCodeImage(HttpServletRequest request, HttpServletResponse response) {
        HttpSession session = request.getSession();
        // BufferedImage类是具有缓冲区的Image类,Image类是用于描述图像信息的类
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_BGR);
        Graphics g = image.getGraphics();
        g.fillRect(0, 0, width, height);
        g.setColor(getRandomColor(105, 189));
        g.setFont(getFont());
        
        // 绘制干扰线
        for (int i = 0; i < lineSize; i++) {
            drawLine(g);
        }
        
        // 绘制随机字符
        String random_string = "";
        for (int i = 0; i < stringNum; i++) {
            random_string = drawString(g, random_string, i);
        }
        
        System.out.println(random_string);
        
        g.dispose();
        
        session.removeAttribute(sessionKey);
        session.setAttribute(sessionKey, random_string);
        
        String base64String = "";
        try {
            //  直接返回图片
           ImageIO.write(image, "PNG", response.getOutputStream()); 
            
        } catch (Exception e) {
            e.printStackTrace();
        }
    }  
    
}

接下来写个 Controller , 提供个接口给前端:

package hello;

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.HashMap;
import java.util.Map;

@RestController
@RequestMapping("/api/v1/user")
public class ValidateCodeController {
        
    
    // 生成验证码图片
    @RequestMapping("/getCaptchaImage")
    @ResponseBody
    public void getCaptcha(HttpServletRequest request, HttpServletResponse response) {
        
        try {
            
            response.setContentType("image/png");
            response.setHeader("Cache-Control", "no-cache");
            response.setHeader("Expire", "0");
            response.setHeader("Pragma", "no-cache");
            
            ValidateCode validateCode = new ValidateCode();
            
            // 直接返回图片
            validateCode.getRandomCodeImage(request, response);
            
        } catch (Exception e) {
            System.out.println(e);
        }
        
    }
    
}

2 前端调取接口

spring Boot手把手教学(11):生成动态验证码并前后端校验 结果如图: spring Boot手把手教学(11):生成动态验证码并前后端校验 spring Boot手把手教学(11):生成动态验证码并前后端校验

3 返回 base64 字符串

有时候我们不能直接返回图片,需要返回一个 json 的数据比如:

spring Boot手把手教学(11):生成动态验证码并前后端校验

这时候我们就需要把 image 转化为 base64

具体代码如下:

在之前的 ValidateCode 类中添加一个方法:

  /*
     *  生成随机图片,返回 base64 字符串
     */
    public String getRandomCodeBase64(HttpServletRequest request, HttpServletResponse response) {
        HttpSession session = request.getSession();
        // BufferedImage类是具有缓冲区的Image类,Image类是用于描述图像信息的类
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_BGR);
        Graphics g = image.getGraphics();
        g.fillRect(0, 0, width, height);
        g.setColor(getRandomColor(105, 189));
        g.setFont(getFont());
        
        // 绘制干扰线
        for (int i = 0; i < lineSize; i++) {
            drawLine(g);
        }
        
        // 绘制随机字符
        String random_string = "";
        for (int i = 0; i < stringNum; i++) {
            random_string = drawString(g, random_string, i);
        }
        
        System.out.println(random_string);
        
        g.dispose();
        
        session.removeAttribute(sessionKey);
        session.setAttribute(sessionKey, random_string);
        
        String base64String = "";
        try {
            //  直接返回图片
            //  ImageIO.write(image, "PNG", response.getOutputStream());
            //返回 base64
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ImageIO.write(image, "PNG", bos);
            
            byte[] bytes = bos.toByteArray();
            Base64.Encoder encoder = Base64.getEncoder();
            base64String = encoder.encodeToString(bytes);
            
        } catch (Exception e) {
            e.printStackTrace();
        }
        
        return base64String;
    }

在 Controller 添加另外一个路由接口:

// 生成验证码,返回的是 base64
    @RequestMapping("/getCaptchaBase64")
    @ResponseBody
    public Object getCaptchaBase64(HttpServletRequest request, HttpServletResponse response) {
        
        Map result = new HashMap();
        Response response1 = new Response();
        
        try {
            
            response.setContentType("image/png");
            response.setHeader("Cache-Control", "no-cache");
            response.setHeader("Expire", "0");
            response.setHeader("Pragma", "no-cache");
            
            ValidateCode validateCode = new ValidateCode();
            
            // 直接返回图片
            // validateCode.getRandomCode(request, response);
            
            // 返回base64
            String base64String = validateCode.getRandomCodeBase64(request, response);
            result.put("url", "data:image/png;base64," + base64String);
            result.put("message", "created successfull");
            System.out.println("test=" + result.get("url"));
            response1.setData(0, result);
            
        } catch (Exception e) {
            System.out.println(e);
        }
        
        return response1.getResult();
    }

调用结果:

spring Boot手把手教学(11):生成动态验证码并前后端校验

spring Boot手把手教学(11):生成动态验证码并前后端校验 在前端页面中,只要把 URL 放到 imageURL 中,即可显示,这里不再演示。

3 验证验证码

生成: spring Boot手把手教学(11):生成动态验证码并前后端校验 验证: spring Boot手把手教学(11):生成动态验证码并前后端校验

本文使用 mdnice 排版

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/13062.html

(0)

相关推荐

  • mysql调优参数有哪些_MySQL调优

    mysql调优参数有哪些_MySQL调优MySQL调优 优化需要考虑哪些方面 优化目标与方向定位 总体目标:使得响应时间更快,吞吐量更大。 (throughout 吞吐量:单位时间内处理事务的数量) 如何找到需要优化的地方 使用反馈。比如做

    2023-03-18
    158
  • Python字典转字符串,精简操作代码

    Python字典转字符串,精简操作代码在Python开发中,常常需要将字典类型的数据转换为字符串类型。文字类型的数据可用于数据存储、网络传输、数据展示等多种场景。Python提供了多种方法进行字典转字符串,但是通常需要进行一些额外处理,使过程变得繁琐。本篇文章将介绍一种精简、高效的Python字典转字符串的方法,省去多余操作。

    2023-12-14
    103
  • SSM项目下Druid连接池的配置及数据源监控的使用[通俗易懂]

    SSM项目下Druid连接池的配置及数据源监控的使用[通俗易懂]一,连接池的配置
    二,数据源监控的配置

    2023-01-29
    155
  • 配置Python在Linux中的环境变量

    配置Python在Linux中的环境变量环境变量是一组在操作系统中定义的动态值,用于控制操作系统及其应用程序的行为。Linux操作系统中,环境变量主要分为系统环境变量和用户环境变量两种。系统环境变量由系统管理员在操作系统中设置,对所有用户生效;用户环境变量则由用户自行设置,只对当前用户生效。

    2024-03-03
    94
  • Hadoop-谈谈你对Hadoop的正确认识和理解

    Hadoop-谈谈你对Hadoop的正确认识和理解一.什么是hadoop? 1.Hadoop是Apache旗下的一套开源软件平台,是用来分析和处理大数据的软件平台。 2.Hadoop提供的功能:利用服务器集群,根据用户的自定义业务逻辑, 对海量数据进

    2023-03-07
    153
  • 使用Python os模块的environ方法设置环境变量

    使用Python os模块的environ方法设置环境变量在操作系统中,环境变量说白了就是存储特定信息的一组变量,是一种全局变量。在Python中,可以使用os模块中的environ方法来设置和修改环境变量,该方法可以通过Python程序设置操作系统的环境变量,从而实现进程间通信、系统配置和软件开发等方面的需求。

    2024-02-12
    90
  • Java不可变ArrayList[通俗易懂]

    Java不可变ArrayList[通俗易懂]集合类在Java中是引用类型,在操作的时候可能不经意间被程序修改,一些初级开发经常会犯类似的错误。这里我们讨论下如何让ArrayList不可改变。本文将演示以下几种方式: JDK Guava Apache Commons Collection4 JDK给我们提供了一个很简便的方…

    2023-07-30
    121
  • 代理《神途》有哪些优势和劣势_代理过神途的心得

    代理《神途》有哪些优势和劣势_代理过神途的心得自2014年起,代理加盟《神途》的人越来越多,那么《神途》究竟有哪些优势,会受广大私服、工作室,甚至是个人玩家所青睐呢?

    2023-06-27
    123

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注