SpringBoot登录验证码实现过程详解

 更新时间:2020年04月10日 08:28:01   作者:晓宇码匠  
这篇文章主要介绍了SpringBoot登录验证码实现过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

今天记录一下验证码的实现,希望能够帮助到大家!

首先我们看一下实现的效果:

此验证码的实现没有用到太多的插件,话不多说直接上代码,大家拿过去就可以用。

中间用到了org.apache.commons.lang3.RandomUtils工具类,需要pom配置:

<!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 -->
    <dependency>
      <groupId>org.apache.commons</groupId>
      <artifactId>commons-lang3</artifactId>
      <version>3.10</version>
    </dependency>

1.验证码类

package com.youyou.login.util.validatecode;

import lombok.Data;

/**
* 验证码类
*/
public class VerifyCode {
  private String code;
  private byte[] imgBytes;
  private long expireTime;
  public String getCode() {
    return code;
  }
  public void setCode(String code) {
    this.code = code;
  }
  public byte[] getImgBytes() {
    return imgBytes;
  }
  public void setImgBytes(byte[] imgBytes) {
    this.imgBytes = imgBytes;
  }
  public long getExpireTime() {
    return expireTime;
  }
  public void setExpireTime(long expireTime) {
    this.expireTime = expireTime;
  }

}

2.验证码生成接口

package com.youyou.login.util.validatecode;

import java.io.IOException;
import java.io.OutputStream;

/**
* 验证码生成接口
*/
public interface IVerifyCodeGen {

/**
* 生成验证码并返回code,将图片写的os中
*
* @param width
* @param height
* @param os
* @return
* @throws IOException
*/
String generate(int width, int height, OutputStream os) throws IOException;

/**
* 生成验证码对象
*
* @param width
* @param height
* @return
* @throws IOException
*/
VerifyCode generate(int width, int height) throws IOException;
}

3.验证码生成实现类

package com.youyou.login.util.validatecode;
import com.youyou.util.RandomUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Random;
/**
* 验证码实现类
*/
public class SimpleCharVerifyCodeGenImpl implements IVerifyCodeGen {
	private static final Logger logger = LoggerFactory.getLogger(SimpleCharVerifyCodeGenImpl.class);
	private static final String[] FONT_TYPES = { "u5b8bu4f53", "u65b0u5b8bu4f53", "u9ed1u4f53", "u6977u4f53", "u96b6u4e66" };
	private static final int VALICATE_CODE_LENGTH = 4;
	/**
* 设置背景颜色及大小,干扰线
*
* @param graphics
* @param width
* @param height
*/
	private static void fillBackground(Graphics graphics, int width, int height) {
		// 填充背景
		graphics.setColor(Color.WHITE);
		//设置矩形坐标x y 为0
		graphics.fillRect(0, 0, width, height);
		// 加入干扰线条
		for (int i = 0; i < 8; i++) {
			//设置随机颜色算法参数
			graphics.setColor(RandomUtils.randomColor(40, 150));
			Random random = new Random();
			int x = random.nextint(width);
			int y = random.nextint(height);
			int x1 = random.nextint(width);
			int y1 = random.nextint(height);
			graphics.drawLine(x, y, x1, y1);
		}
	}
	/**
* 生成随机字符
*
* @param width
* @param height
* @param os
* @return
* @throws IOException
*/
	@Override
	public String generate(int width, int height, OutputStream os) throws IOException {
		BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
		Graphics graphics = image.getGraphics();
		fillBackground(graphics, width, height);
		String randomStr = RandomUtils.randomString(VALICATE_CODE_LENGTH);
		createCharacter(graphics, randomStr);
		graphics.dispose();
		//设置JPEG格式
		ImageIO.write(image, "JPEG", os);
		return randomStr;
	}
	/**
* 验证码生成
*
* @param width
* @param height
* @return
*/
	@Override
	public VerifyCode generate(int width, int height) {
		VerifyCode verifyCode = null;
		try (
		//将流的初始化放到这里就不需要手动关闭流
		ByteArrayOutputStream baos = new ByteArrayOutputStream();
		) {
			String code = generate(width, height, baos);
			verifyCode = new VerifyCode();
			verifyCode.setCode(code);
			verifyCode.setImgBytes(baos.toByteArray());
		}
		catch (IOException e) {
			logger.error(e.getMessage(), e);
			verifyCode = null;
		}
		return verifyCode;
	}
	/**
* 设置字符颜色大小
*
* @param g
* @param randomStr
*/
	private void createCharacter(Graphics g, String randomStr) {
		char[] charArray = randomStr.toCharArray();
		for (int i = 0; i < charArray.length; i++) {
			//设置RGB颜色算法参数
			g.setColor(new Color(50 + RandomUtils.nextint(100),
			+ RandomUtils.nextint(100), 50 + RandomUtils.nextint(100)));
			//设置字体大小,类型
			g.setFont(new Font(FONT_TYPES[RandomUtils.nextint(FONT_TYPES.length)], Font.BOLD, 26));
			//设置x y 坐标
			g.drawString(String.valueOf(charArray[i]), 15 * i + 5, 19 + RandomUtils.nextint(8));
		}
	}
}

4.工具类

package com.youyou.util;
import java.awt.*;
import java.util.Random;
public class RandomUtils extends org.apache.commons.lang3.RandomUtils {
	private static final char[] CODE_SEQ = { 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'J',
	'K', 'L', 'M', 'N', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',
	'X', 'Y', 'Z', '2', '3', '4', '5', '6', '7', '8', '9' };
	private static final char[] NUMBER_ARRAY = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' };
	private static Random random = new Random();
	public static String randomString(int length) {
		StringBuilder sb = new StringBuilder();
		for (int i = 0; i < length; i++) {
			sb.append(String.valueOf(CODE_SEQ[random.nextint(CODE_SEQ.length)]));
		}
		return sb.toString();
	}
	public static String randomNumberString(int length) {
		StringBuilder sb = new StringBuilder();
		for (int i = 0; i < length; i++) {
			sb.append(String.valueOf(NUMBER_ARRAY[random.nextint(NUMBER_ARRAY.length)]));
		}
		return sb.toString();
	}
	public static Color randomColor(int fc, int bc) {
		int f = fc;
		int b = bc;
		Random random = new Random();
		if (f > 255) {
			f = 255;
		}
		if (b > 255) {
			b = 255;
		}
		return new Color(f + random.nextint(b - f), f + random.nextint(b - f), f + random.nextint(b - f));
	}
	public static int nextint(int bound) {
		return random.nextint(bound);
	}
}

经过以上代码,我们的验证码生成功能基本上已经实现了,现在还需要一个controller来调用它。

@ApiOperation(value = "验证码")
@GetMapping("/verifyCode")
public void verifyCode(HttpServletRequest request, HttpServletResponse response) {
	IVerifyCodeGen iVerifyCodeGen = new SimpleCharVerifyCodeGenImpl();
	try {
		//设置长宽
		VerifyCode verifyCode = iVerifyCodeGen.generate(80, 28);
		String code = verifyCode.getCode();
		LOGGER.info(code);
		//将VerifyCode绑定session
		request.getSession().setAttribute("VerifyCode", code);
		//设置响应头
		response.setHeader("Pragma", "no-cache");
		//设置响应头
		response.setHeader("Cache-Control", "no-cache");
		//在代理服务器端防止缓冲
		response.setDateHeader("Expires", 0);
		//设置响应内容类型
		response.setContentType("image/jpeg");
		response.getOutputStream().write(verifyCode.getImgBytes());
		response.getOutputStream().flush();
	}
	catch (IOException e) {
		LOGGER.info("", e);
	}
}

搞定!后台编写到此结束了。那么又会有博友说了:“说好的实现效果呢?”

好吧,那么我们继续前端的代码编写。

前端代码:

<html>
<body>

<div>
<input id="code" placeholder="验证码" type="text" class=""
style="width:170px">
<!-- 验证码 显示 -->
<img οnclick="javascript:getvCode()" id="verifyimg" style="margin-left: 20px;"/>
</div>

<script type="text/javascript">
getvCode();

/**
* 获取验证码
* 将验证码写到login.html页面的id = verifyimg 的地方
*/
function getvCode() {
document.getElementById("verifyimg").src = timestamp("http://127.0.0.1:81/verifyCode");
}
//为url添加时间戳
function timestamp(url) {
var getTimestamp = new Date().getTime();
if (url.indexOf("?") > -1) {
url = url + "&timestamp=" + getTimestamp
} else {
url = url + "?timestamp=" + getTimestamp
}
return url;
};
</script>
</body>

</html>

可以实现点击图片更换验证码。

实现效果:

当然文章开头的截图是我系统中的截图,需要大家自己去根据自己的情况去开发前端了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Java 常量池详解之class文件常量池 和class运行时常量池

    Java 常量池详解之class文件常量池 和class运行时常量池

    这篇文章主要介绍了Java 常量池详解之class文件常量池 和class运行时常量池,常量池主要存放两大类常量:字面量,符号引用,本文结合示例代码对java class常量池相关知识介绍的非常详细,需要的朋友可以参考下
    2022-12-12
  • Java 中二进制转换成十六进制的两种实现方法

    Java 中二进制转换成十六进制的两种实现方法

    这篇文章主要介绍了Java 中二进制转换成十六进制的两种实现方法的相关资料,需要的朋友可以参考下
    2017-06-06
  • 使用Java实现2048小游戏代码实例

    使用Java实现2048小游戏代码实例

    这篇文章主要介绍了使用Java实现2048小游戏代码实例,2048 游戏是一款益智类游戏,玩家需要通过合并相同数字的方块,不断合成更大的数字,最终达到2048,游戏规则简单,但挑战性很高,需要玩家灵活运用策略和计算能力,本文将使用Java代码实现,需要的朋友可以参考下
    2023-10-10
  • MyBatis Mapper.XML 标签使用小结

    MyBatis Mapper.XML 标签使用小结

    在MyBatis中,通过resultMap可以解决字段名和属性名不一致的问题,对于复杂的查询,引用实体或使用<sql>标签可以定义复用的SQL片段,提高代码的可读性和编码效率,使用这些高级映射和动态SQL技巧,可以有效地处理复杂的数据库交互场景
    2024-10-10
  • 详解Spring如何更简单的读取和存储对象

    详解Spring如何更简单的读取和存储对象

    这篇文章主要为大家详细介绍了Spring中如何更简单的实现读取和存储对象,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-07-07
  • java 基础教程之多线程详解及简单实例

    java 基础教程之多线程详解及简单实例

    这篇文章主要介绍了java 基础教程之多线程详解及简单实例的相关资料,线程的基本属性、如何创建线程、线程的状态切换以及线程通信,需要的朋友可以参考下
    2017-03-03
  • 很多人竟然不知道Java线程池的创建方式有7种

    很多人竟然不知道Java线程池的创建方式有7种

    本文主要介绍了很多人竟然不知道Java线程池的创建方式有7种,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • Java通过正则表达式获取字符串中数字的方法示例

    Java通过正则表达式获取字符串中数字的方法示例

    最近工作中遇到了一个需求,需要利用java获取字符串中的数字,尝试几种方法后发现利用正则表达式实现最为方法,下面这篇文章就主要介绍了Java通过正则表达式获取字符串中数字的方法,文中给出了详细的示例代码,需要的朋友可以参考下。
    2017-03-03
  • Kotlin 基础教程之反射

    Kotlin 基础教程之反射

    这篇文章主要介绍了Kotlin 基础教程之反射的相关资料,需要的朋友可以参考下
    2017-06-06
  • Java调用Zookeeper的实现步骤

    Java调用Zookeeper的实现步骤

    本文主要介绍了Java调用Zookeeper的实现步骤,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08

最新评论