什么專業(yè)可以做網(wǎng)站百度店鋪免費入駐
🌟 前言
歡迎來到我的技術(shù)小宇宙!🌌 這里不僅是我記錄技術(shù)點滴的后花園,也是我分享學(xué)習(xí)心得和項目經(jīng)驗的樂園。📚 無論你是技術(shù)小白還是資深大牛,這里總有一些內(nèi)容能觸動你的好奇心。🔍
🤖 洛可可白:個人主頁
🔥 個人專欄:?前端技術(shù) ?后端技術(shù)
🏠 個人博客:洛可可白博客
🐱 代碼獲取:bestwishes0203
📷 封面壁紙:洛可可白wallpaper
文章目錄
- Spring Boot工程集成驗證碼生成與驗證功能教程
- 1. 創(chuàng)建驗證碼工具類
- 2. 控制層實現(xiàn)
- 3. 客戶端展示驗證碼
- 4. 驗證驗證碼
- 📌 聯(lián)系方式
- 🚀 獲取源代碼
- 🎉 結(jié)語
Spring Boot工程集成驗證碼生成與驗證功能教程
驗證碼是一種常見的安全機制,用于防止自動化工具(如爬蟲)對網(wǎng)站進行惡意操作。在Web應(yīng)用中,驗證碼通常以圖像的形式出現(xiàn),要求用戶輸入圖像中顯示的字符。本文將介紹如何在Spring Boot工程中實現(xiàn)一個隨機生成驗證碼的功能。
1. 創(chuàng)建驗證碼工具類
首先,我們需要創(chuàng)建一個工具類VerifyCodeUtils
,用于生成隨機驗證碼并輸出為圖像。
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Arrays;
import java.util.Random;public class VerifyCodeUtils {//使用到Algerian字體,系統(tǒng)里沒有的話需要安裝字體,字體只顯示大寫,去掉了1,0,i,o幾個容易混淆的字符public static final String VERIFY_CODES = "23456789ABCDEFGHJKLMNPQRSTUVWXYZ";private static Random random = new Random();/*** 使用系統(tǒng)默認字符源生成驗證碼** @param verifySize 驗證碼長度* @return*/public static String generateVerifyCode(int verifySize) {return generateVerifyCode(verifySize, VERIFY_CODES);}/*** 使用指定源生成驗證碼** @param verifySize 驗證碼長度* @param sources 驗證碼字符源* @return*/public static String generateVerifyCode(int verifySize, String sources) {if (sources == null || sources.length() == 0) {sources = VERIFY_CODES;}int codesLen = sources.length();Random rand = new Random(System.currentTimeMillis());StringBuilder verifyCode = new StringBuilder(verifySize);for (int i = 0; i < verifySize; i++) {verifyCode.append(sources.charAt(rand.nextInt(codesLen - 1)));}return verifyCode.toString();}/*** 生成隨機驗證碼文件,并返回驗證碼值** @param w* @param h* @param outputFile* @param verifySize* @return* @throws IOException*/public static String outputVerifyImage(int w, int h, File outputFile, int verifySize) throws IOException {String verifyCode = generateVerifyCode(verifySize);outputImage(w, h, outputFile, verifyCode);return verifyCode;}/*** 輸出隨機驗證碼圖片流,并返回驗證碼值** @param w* @param h* @param os* @param verifySize* @return* @throws IOException*/public static String outputVerifyImage(int w, int h, OutputStream os, int verifySize) throws IOException {String verifyCode = generateVerifyCode(verifySize);outputImage(w, h, os, verifyCode);return verifyCode;}/*** 生成指定驗證碼圖像文件** @param w* @param h* @param outputFile* @param code* @throws IOException*/public static void outputImage(int w, int h, File outputFile, String code) throws IOException {if (outputFile == null) {return;}File dir = outputFile.getParentFile();if (!dir.exists()) {dir.mkdirs();}try {outputFile.createNewFile();FileOutputStream fos = new FileOutputStream(outputFile);outputImage(w, h, fos, code);fos.close();} catch (IOException e) {throw e;}}/*** 輸出指定驗證碼圖片流** @param w* @param h* @param os* @param code* @throws IOException*/public static void outputImage(int w, int h, OutputStream os, String code) throws IOException {int verifySize = code.length();BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);Random rand = new Random();Graphics2D g2 = image.createGraphics();g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);Color[] colors = new Color[5];Color[] colorSpaces = new Color[]{Color.WHITE, Color.CYAN,Color.GRAY, Color.LIGHT_GRAY, Color.MAGENTA, Color.ORANGE,Color.PINK, Color.YELLOW};float[] fractions = new float[colors.length];for (int i = 0; i < colors.length; i++) {colors[i] = colorSpaces[rand.nextInt(colorSpaces.length)];fractions[i] = rand.nextFloat();}Arrays.sort(fractions);g2.setColor(Color.GRAY);// 設(shè)置邊框色g2.fillRect(0, 0, w, h);Color c = getRandColor(200, 250);g2.setColor(c);// 設(shè)置背景色g2.fillRect(0, 2, w, h - 4);//繪制干擾線Random random = new Random();g2.setColor(getRandColor(160, 200));// 設(shè)置線條的顏色for (int i = 0; i < 20; i++) {int x = random.nextInt(w - 1);int y = random.nextInt(h - 1);int xl = random.nextInt(6) + 1;int yl = random.nextInt(12) + 1;g2.drawLine(x, y, x + xl + 40, y + yl + 20);}// 添加噪點float yawpRate = 0.05f;// 噪聲率int area = (int) (yawpRate * w * h);for (int i = 0; i < area; i++) {int x = random.nextInt(w);int y = random.nextInt(h);int rgb = getRandomIntColor();image.setRGB(x, y, rgb);}shear(g2, w, h, c);// 使圖片扭曲g2.setColor(getRandColor(100, 160));int fontSize = h - 4;Font font = new Font("Algerian", Font.ITALIC, fontSize);g2.setFont(font);char[] chars = code.toCharArray();for (int i = 0; i < verifySize; i++) {AffineTransform affine = new AffineTransform();affine.setToRotation(Math.PI / 4 * rand.nextDouble() * (rand.nextBoolean() ? 1 : -1), (w / verifySize) * i + fontSize / 2, h / 2);g2.setTransform(affine);g2.drawChars(chars, i, 1, ((w - 10) / verifySize) * i + 5, h / 2 + fontSize / 2 - 10);}g2.dispose();ImageIO.write(image, "jpg", os);}private static Color getRandColor(int fc, int bc) {if (fc > 255)fc = 255;if (bc > 255)bc = 255;int r = fc + random.nextInt(bc - fc);int g = fc + random.nextInt(bc - fc);int b = fc + random.nextInt(bc - fc);return new Color(r, g, b);}private static int getRandomIntColor() {int[] rgb = getRandomRgb();int color = 0;for (int c : rgb) {color = color << 8;color = color | c;}return color;}private static int[] getRandomRgb() {int[] rgb = new int[3];for (int i = 0; i < 3; i++) {rgb[i] = random.nextInt(255);}return rgb;}private static void shear(Graphics g, int w1, int h1, Color color) {shearX(g, w1, h1, color);shearY(g, w1, h1, color);}private static void shearX(Graphics g, int w1, int h1, Color color) {int period = random.nextInt(2);boolean borderGap = true;int frames = 1;int phase = random.nextInt(2);for (int i = 0; i < h1; i++) {double d = (double) (period >> 1)* Math.sin((double) i / (double) period+ (6.2831853071795862D * (double) phase)/ (double) frames);g.copyArea(0, i, w1, 1, (int) d, 0);if (borderGap) {g.setColor(color);g.drawLine((int) d, i, 0, i);g.drawLine((int) d + w1, i, w1, i);}}}private static void shearY(Graphics g, int w1, int h1, Color color) {int period = random.nextInt(40) + 10; // 50;boolean borderGap = true;int frames = 20;int phase = 7;for (int i = 0; i < w1; i++) {double d = (double) (period >> 1)* Math.sin((double) i / (double) period+ (6.2831853071795862D * (double) phase)/ (double) frames);g.copyArea(i, 0, 1, h1, 0, (int) d);if (borderGap) {g.setColor(color);g.drawLine(i, (int) d, i, 0);g.drawLine(i, (int) d + h1, i, h1);}}}
}
2. 控制層實現(xiàn)
在Spring Boot的控制器中,我們需要提供一個接口來生成驗證碼并將其發(fā)送給客戶端。
// VerifyController.java
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
public class VerifyController {@GetMapping("/generateImageCode")public void generateImageCode(HttpSession session, HttpServletResponse response) throws IOException {//隨機生成四位隨機數(shù)String code = VerifyCodeUtils.generateVerifyCode(4);//保存到session域中session.setAttribute("code", code);//根據(jù)隨機數(shù)生成圖片,reqponse響應(yīng)圖片response.setContentType("image/png");ServletOutputStream os = response.getOutputStream();VerifyCodeUtils.outputImage(130, 60, os, code);}
}
3. 客戶端展示驗證碼
在Web頁面中,我們需要添加一個圖像標簽來展示驗證碼。
<!-- 在HTML中添加驗證碼圖像 -->
<img src="/generateImageCode" alt="驗證碼" onclick="this.src='/generateImageCode?'+Math.random();" />
4. 驗證驗證碼
在用戶提交表單時,我們需要驗證用戶輸入的驗證碼是否正確。這通常在后端進行,通過比較用戶輸入的驗證碼與Session中保存的驗證碼。
// 在登錄方法中驗證驗證碼
@PostMapping("/login")
public ResponseEntity<?> login(@RequestParam String username, @RequestParam String password, HttpSession session) {// 獲取用戶輸入的驗證碼String userInputCode = request.getParameter("code");// 獲取Session中保存的驗證碼String sessionCode = (String) session.getAttribute("code");// 驗證驗證碼if (!sessionCode.equals(inputCode)) {// 驗證碼錯誤return ResponseEntity.status(HttpStatus.BAD_REQUEST).body("驗證碼錯誤");}// 驗證碼正確,繼續(xù)登錄邏輯// ...
}
📌 聯(lián)系方式
如果您對我們的項目感興趣,或者有任何技術(shù)問題想要探討,歡迎通過以下方式與我聯(lián)系。我非常期待與您交流,共同學(xué)習(xí),共同進步!🌊💡🤖
- 郵箱:2109664977@qq.com 📧
- Gitee:https://gitee.com/bestwishes0203 🐱
- GitHub:https://github.com/bestwishes0203 🐙
- CSDN主頁:https://blog.csdn.net/interest_ing_/ 📖
- 個人博客:訪問我的博客 🏠
🚀 獲取源代碼
- 后端案例:https://gitee.com/bestwishes0203/Front-end-example
- 前端案例:https://gitee.com/bestwishes0203/Back-end-example
🎉 結(jié)語
感謝你的訪問,如果你對我的技術(shù)文章或項目感興趣,歡迎通過以上方式與我聯(lián)系。讓我們一起在技術(shù)的道路上不斷前行!🚀