首頁 > 軟體

Java實現動態驗證碼生成

2022-04-22 10:01:05

本文範例為大家分享了Java動態驗證碼生成的具體程式碼,供大家參考,具體內容如下

說明:今天給大家來帶來一個自動生成驗證碼的處理方法。驗證碼的出現有效減少了注入灌水以及破解密碼等惡意操作,提高了系統執行的流暢性,保護了系統使用者的隱私安全,具體實現方法如下:

1.首先我們先編寫一個專門的驗證碼生成工具類,該工具類程式碼如下:

package com.ordering.util;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.util.Random;

/**
 * 驗證碼生成器
 * @author 諾坎普10號
 * @date 2020-2-25
 */
public class CpachaUtil {
    
    /**
     * 驗證碼來源
     */
    final private char[] code = {
        '2', '3', '4', '5', '6', '7', '8', '9',
        'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j',
        'k', 'm', 'n', 'p', 'q', 'r', 's', 't', 'u', 'v', 
        'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F',
        'G', 'H', 'J', 'K', 'L', 'M', 'N', 'P', 'Q', 'R',
        'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'
    };
    /**
     * 字型
     */
    final private String[] fontNames = new String[]{
            "黑體", "宋體", "Courier", "Arial", 
            "Verdana", "Times", "Tahoma", "Georgia"};
    /**
     * 字型樣式
     */
    final private int[] fontStyles = new int[]{
            Font.BOLD, Font.ITALIC|Font.BOLD
    };
    
    /**
     * 驗證碼長度
     * 預設4個字元
     */
    private int vcodeLen = 4;
    /**
     * 驗證碼圖片字型大小
     * 預設17
     */
    private int fontsize = 21;
    /**
     * 驗證碼圖片寬度
     */
    private int width = (fontsize+1)*vcodeLen+10;
    /**
     * 驗證碼圖片高度
     */
    private int height = fontsize+12;
    /**
     * 干擾線條數
     * 預設3條
     */
    private int disturbline = 3;
    
    
    public CpachaUtil(){}
    
    /**
     * 指定驗證碼長度
     * @param vcodeLen 驗證碼長度
     */
    public CpachaUtil(int vcodeLen) {
        this.vcodeLen = vcodeLen;
        this.width = (fontsize+1)*vcodeLen+10;
    }
    
    /**
     * 指定驗證碼長度、圖片寬度、高度
     * @param vcodeLen
     * @param width
     * @param height
     */
    public CpachaUtil(int vcodeLen,int width,int height) {
        this.vcodeLen = vcodeLen;
        this.width = width;
        this.height = height;
    }
    
    /**
     * 生成驗證碼圖片
     * @param vcode 要畫的驗證碼
     * @param drawline 是否畫干擾線
     * @return
     */
    public BufferedImage generatorVCodeImage(String vcode, boolean drawline){
        //建立驗證碼圖片
        BufferedImage vcodeImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
        Graphics g = vcodeImage.getGraphics();
        //填充背景色
        g.setColor(new Color(246, 240, 250));
        g.fillRect(0, 0, width, height);
        if(drawline){
            drawDisturbLine(g);
        }
        //用於生成偽亂數
        Random ran = new Random();
        //在圖片上畫驗證碼
        for(int i = 0;i < vcode.length();i++){
            //設定字型
            g.setFont(new Font(fontNames[ran.nextInt(fontNames.length)], fontStyles[ran.nextInt(fontStyles.length)], fontsize));
            //隨機生成顏色
            g.setColor(getRandomColor());
            //畫驗證碼
            g.drawString(vcode.charAt(i)+"", i*fontsize+10, fontsize+5);
        }
        //釋放此圖形的上下文以及它使用的所有系統資源
        g.dispose();
        
        return vcodeImage;
    }
    /**
     * 獲得旋轉字型的驗證碼圖片
     * @param vcode
     * @param drawline 是否畫干擾線
     * @return
     */
    public BufferedImage generatorRotateVCodeImage(String vcode, boolean drawline){
        //建立驗證碼圖片
        BufferedImage rotateVcodeImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
        Graphics2D g2d = rotateVcodeImage.createGraphics();
        //填充背景色
        g2d.setColor(new Color(246, 240, 250));
        g2d.fillRect(0, 0, width, height);
        if(drawline){
            drawDisturbLine(g2d);
        }
        //在圖片上畫驗證碼
        for(int i = 0;i < vcode.length();i++){
            BufferedImage rotateImage = getRotateImage(vcode.charAt(i));
            g2d.drawImage(rotateImage, null, (int) (this.height * 0.7) * i, 0);
        }
        g2d.dispose();
        return rotateVcodeImage;
    }
    /**
     * 生成驗證碼
     * @return 驗證碼
     */
    public String generatorVCode(){
        int len = code.length;
        Random ran = new Random();
        StringBuffer sb = new StringBuffer();
        for(int i = 0;i < vcodeLen;i++){
            int index = ran.nextInt(len);
            sb.append(code[index]);
        }
        return sb.toString();
    }
    /**
     * 為驗證碼圖片畫一些干擾線
     * @param g 
     */
    private void drawDisturbLine(Graphics g){
        Random ran = new Random();
        for(int i = 0;i < disturbline;i++){
            int x1 = ran.nextInt(width);
            int y1 = ran.nextInt(height);
            int x2 = ran.nextInt(width);
            int y2 = ran.nextInt(height);
            g.setColor(getRandomColor());
            //畫干擾線
            g.drawLine(x1, y1, x2, y2);
        }
    }
    /**
     * 獲取一張旋轉的圖片
     * @param c 要畫的字元
     * @return
     */
    private BufferedImage getRotateImage(char c){
        BufferedImage rotateImage = new BufferedImage(height, height, BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2d = rotateImage.createGraphics();
        //設定透明度為0
        g2d.setColor(new Color(255, 255, 255, 0));
        g2d.fillRect(0, 0, height, height);
        Random ran = new Random();
        g2d.setFont(new Font(fontNames[ran.nextInt(fontNames.length)], fontStyles[ran.nextInt(fontStyles.length)], fontsize));
        g2d.setColor(getRandomColor());
        double theta = getTheta();
        //旋轉圖片
        g2d.rotate(theta, height/2, height/2);
        g2d.drawString(Character.toString(c), (height-fontsize)/2, fontsize+5);
        g2d.dispose();
        
        return rotateImage;
    }
    /**
     * @return 返回一個隨機顏色
     */
    private Color getRandomColor(){
        Random ran = new Random();
        return new Color(ran.nextInt(220), ran.nextInt(220), ran.nextInt(220)); 
    }
    /**
     * @return 角度
     */
    private double getTheta(){
        return ((int) (Math.random()*1000) % 2 == 0 ? -1 : 1)*Math.random();
    }

    /**
     * @return 驗證碼字元個數
     */
    public int getVcodeLen() {
        return vcodeLen;
    }
    /**
     * 設定驗證碼字元個數
     * @param vcodeLen
     */
    public void setVcodeLen(int vcodeLen) {
        this.width = (fontsize+3)*vcodeLen+10;
        this.vcodeLen = vcodeLen;
    }
    /**
     * @return 字型大小
     */
    public int getFontsize() {
        return fontsize;
    }
    /**
     * 設定字型大小
     * @param fontsize
     */
    public void setFontsize(int fontsize) {
        this.width = (fontsize+3)*vcodeLen+10;
        this.height = fontsize+15;
        this.fontsize = fontsize;
    }
    /**
     * @return 圖片寬度
     */
    public int getWidth() {
        return width;
    }
    /**
     * 設定圖片寬度
     * @param width
     */
    public void setWidth(int width) {
        this.width = width;
    }
    /**
     * @return 圖片高度
     */
    public int getHeight() {
        return height;
    }
    /**
     * 設定圖片高度
     * @param height 
     */
    public void setHeight(int height) {
        this.height = height;
    }
    /**
     * @return 干擾線條數
     */
    public int getDisturbline() {
        return disturbline;
    }
    /**
     * 設定干擾線條數
     * @param disturbline
     */
    public void setDisturbline(int disturbline) {
        this.disturbline = disturbline;
    }
    
}

2.然後在頁面端定義一個顯示驗證碼的控制元件,再寫一個js函數來對後臺動態生成驗證碼方法進行呼叫,主要程式碼如下:

<div class="login-center-input-text">驗證碼</div>
<img id="cpacha-img" title="點選切換驗證碼" style="cursor:pointer;" src="get_cpacha?vl=4&w=150&h=40&type=loginCpacha" width="110px" height="30px" onclick="changeCpacha()">
</div>

<script type="text/javascript">
function changeCpacha(){
        $("#cpacha-img").attr("src",'get_cpacha?vl=4&w=150&h=40&type=loginCpacha&t=' + new Date().getTime());
    }
</script>

3.控制器中程式碼如下:

/**
     * 本系統所有的驗證碼均採用此方法
     * 
     * @param vcodeLen
     * @param width
     * @param height
     * @param cpachaType:用來區別驗證碼的型別,傳入字串
     * @param request
     * @param response
     */
    @RequestMapping(value = "/get_cpacha", method = RequestMethod.GET)
    public void generateCpacha(@RequestParam(name = "vl", required = false, defaultValue = "4") Integer vcodeLen,
            @RequestParam(name = "w", required = false, defaultValue = "100") Integer width,
            @RequestParam(name = "h", required = false, defaultValue = "30") Integer height,
            @RequestParam(name = "type", required = true, defaultValue = "loginCpacha") String cpachaType,
            HttpServletRequest request, HttpServletResponse response) {
        //根據頁面傳入的驗證碼長度、圖片寬度以及圖片長度來範例化驗證碼實體
        CpachaUtil cpachaUtil = new CpachaUtil(vcodeLen, width, height);
        //生成驗證碼
        String generatorVCode = cpachaUtil.generatorVCode();
        //把驗證碼存入session
        request.getSession().setAttribute(cpachaType, generatorVCode);
        //根據驗證碼生成相應的驗證碼圖片
        BufferedImage generatorRotateVCodeImage = cpachaUtil.generatorRotateVCodeImage(generatorVCode, true);
        try {
            ImageIO.write(generatorRotateVCodeImage, "gif", response.getOutputStream());
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

4.實現效果如下圖所示:

5.控制器驗證碼驗證部分如下圖所示:

//若驗證碼為空
if (StringUtils.isEmpty(cpacha)) {
            ret.put("type", "error");
            ret.put("msg", "請填寫驗證碼!");
            return ret;
        }
        //獲取儲存在session中的登入驗證碼物件
        Object loginCpacha = request.getSession().getAttribute("loginCpacha");
        //若登入驗證碼實體物件為空
        if (loginCpacha == null) {
            ret.put("type", "error");
            ret.put("msg", "對談超時,請重新整理頁面!");
            return ret;
        }
        //若輸入的驗證碼轉化成大寫之後不等於轉換成大寫的session中儲存的驗證碼
        if (!cpacha.toUpperCase().equals(loginCpacha.toString().toUpperCase())) {
            ret.put("type", "error");
            ret.put("msg", "驗證碼錯誤!");
            //範例化紀錄檔資訊實體
            OrderingLog orderingLog = new OrderingLog("使用者名稱為" + user.getOuUsername() + "的使用者登入時輸入驗證碼錯誤!", new Date());
            //新增紀錄檔資訊
            logService.addLog(orderingLog);
            return ret;
        }

注意:

只有當你把當前系統時間傳入該驗證碼生成工具類中,才能實現點選驗證碼圖片,實時自動替換驗證碼,如下圖所示:

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com