java實現(xiàn)隨機生成驗證碼圖片
驗證碼
驗證碼(CAPTCHA)是“Completely Automated Public Turing test to tell Computers and Humans Apart”(全自動區(qū)分計算機和人類的圖靈測試)的縮寫。由于計算機無法解答CAPTCHA的問題,所以回答出問題的用戶就可以被認為是人類。
作用
防止惡意破解密碼、刷票、論壇灌水、刷頁。
有效防止某個黑客對某一個特定注冊用戶用特定程序暴力破解方式進行不斷的登錄嘗試,實際上使用驗證碼是現(xiàn)在很多網(wǎng)站通行的方式(比如招商銀行的網(wǎng)上個人銀行,百度社區(qū)),我們利用比較簡易的方式實現(xiàn)了這個功能。雖然登錄麻煩一點,但是對網(wǎng)友的密碼安全來說這個功能還是很有必要,也很重要。但我們還是 提醒大家要保護好自己的密碼 ,盡量使用混雜了數(shù)字、字母、符號在內(nèi)的6位以上密碼,不要使用諸如1234之類的簡單密碼或者與用戶名相同、類似的密碼 ,免得你的賬號被人盜用給自己帶來不必要的麻煩。
驗證碼通常使用一些線條和一些不規(guī)則的字符組成,主要作用是為了防止一些黑客把密碼數(shù)據(jù)化盜取。
分類
現(xiàn)在常見的驗證碼主要有
- gif驗證碼
- 手機短信驗證碼
- 手機語音驗證碼
- 視頻驗證碼
實例
這里只是用java生出一個驗證碼的圖片,并沒有在在網(wǎng)頁的操作
package com.xn;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Date;
import java.util.Random;
/**
* 驗證碼生成工具類
*
* @leo
*/
public class ValidateCode {
// 圖片的寬度。
private int width = 160;
// 圖片的高度。
private int height = 40;
// 驗證碼字符個數(shù)
private int codeCount = 5;
// 驗證碼干擾線數(shù)
private int lineCount = 150;
// 驗證碼
private String code = null;
// 驗證碼圖片Buffer
private BufferedImage buffImg = null;
// 驗證碼范圍,去掉0(數(shù)字)和O(拼音)容易混淆的(小寫的1和L也可以去掉,大寫不用了)
private char[] codeSequence = {'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
'K', 'L', 'M', 'N', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',
'X', 'Y', 'Z', '1', '2', '3', '4', '5', '6', '7', '8', '9'};
/**
* 默認構(gòu)造函數(shù),設置默認參數(shù)
*/
public ValidateCode() {
this.createCode();
}
/**
* @param width 圖片寬
* @param height 圖片高
*/
public ValidateCode(int width, int height) {
this.width = width;
this.height = height;
this.createCode();
}
/**
* @param width 圖片寬
* @param height 圖片高
* @param codeCount 字符個數(shù)
* @param lineCount 干擾線條數(shù)
*/
public ValidateCode(int width, int height, int codeCount, int lineCount) {
this.width = width;
this.height = height;
this.codeCount = codeCount;
this.lineCount = lineCount;
this.createCode();
}
public void createCode() {
int x = 0, fontHeight = 0, codeY = 0;
int red = 0, green = 0, blue = 0;
x = width / (codeCount + 2);//每個字符的寬度(左右各空出一個字符)
fontHeight = height - 2;//字體的高度
codeY = height - 4;
// 圖像buffer
buffImg = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
Graphics2D g = buffImg.createGraphics();
/*// 將圖像背景填充為白色
g.setColor(Color.WHITE);
g.fillRect(0, 0, width, height);*/
// 增加下面代碼使得背景透明
buffImg = g.getDeviceConfiguration().createCompatibleImage(width, height, Transparency.TRANSLUCENT);
g.dispose();
g = buffImg.createGraphics();
// 背景透明代碼結(jié)束
// 畫圖BasicStroke是JDK中提供的一個基本的畫筆類,我們對他設置畫筆的粗細,就可以在drawPanel上任意畫出自己想要的圖形了。
g.setColor(new Color(255, 0, 0));
g.setStroke(new BasicStroke(1f));
g.fillRect(128, 128, width, height);
// 生成隨機數(shù)
Random random = new Random();
//設置字體類型、字體大小、字體樣式
Font font = new Font("微軟雅黑",Font.PLAIN, fontHeight);
g.setFont(font);
for (int i = 0; i < lineCount; i++) {
// 設置隨機開始和結(jié)束坐標
int xs = random.nextInt(width);//x坐標開始
int ys = random.nextInt(height);//y坐標開始
int xe = xs + random.nextInt(width / 8);//x坐標結(jié)束
int ye = ys + random.nextInt(height / 8);//y坐標結(jié)束
// 產(chǎn)生隨機的顏色值,讓輸出的每個干擾線的顏色值都將不同。
red = random.nextInt(255);
green = random.nextInt(255);
blue = random.nextInt(255);
g.setColor(new Color(red, green, blue));
g.drawLine(xs, ys, xe, ye);
}
// randomCode記錄隨機產(chǎn)生的驗證碼
StringBuffer randomCode = new StringBuffer();
// 隨機產(chǎn)生codeCount個字符的驗證碼。
for (int i = 0; i < codeCount; i++) {
String strRand = String.valueOf(codeSequence[random.nextInt(codeSequence.length)]);
// 產(chǎn)生隨機的顏色值,讓輸出的每個字符的顏色值都將不同。
red = random.nextInt(255);
green = random.nextInt(255);
blue = random.nextInt(255);
//指定某種顏色
//g.setColor(new Color(252, 145, 83));
g.setColor(new Color(red, green, blue));
g.drawString(strRand, (i + 1) * x, codeY);
// 將產(chǎn)生的四個隨機數(shù)組合在一起。
randomCode.append(strRand);
}
// 將四位數(shù)字的驗證碼保存到Session中。
code = randomCode.toString();
}
public void write(String path) throws IOException {
OutputStream sos = new FileOutputStream(path);
this.write(sos);
}
public void write(OutputStream sos) throws IOException {
ImageIO.write(buffImg, "png", sos);
sos.close();
}
public BufferedImage getBuffImg() {
return buffImg;
}
public String getCode() {
return code;
}
/**
* 測試函數(shù),默認生成到d盤
* @param args
*/
public static void main(String[] args) {
ValidateCode vCode = new ValidateCode(160,40,5,150);
try {
String path="D:/"+new Date().getTime()+".png";
System.out.println(vCode.getCode()+" >"+path);
vCode.write(path);
} catch (IOException e) {
e.printStackTrace();
}
}
}
測試:


更多關于驗證碼的文章請點擊查看: 《java驗證碼》
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
POI XSSFSheet shiftRows bug問題解決
這篇文章主要介紹了POI XSSFSheet shiftRows bug問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07
Java 內(nèi)存模型中的happen-before關系詳解
這篇文章主要為大家介紹了Java 內(nèi)存模型中的happen-before關系示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-10-10
IDEA快速部署Spring?Boot?項目到Docker的實現(xiàn)方法
本文主要介紹了IDEA快速部署Spring?Boot?項目到Docker的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-07-07
MyBatis-Plus通過插件將數(shù)據(jù)庫表生成Entiry,Mapper.xml,Mapper.class的方式
今天小編就為大家分享一篇關于MyBatis-Plus通過插件將數(shù)據(jù)庫表生成Entiry,Mapper.xml,Mapper.class的方式,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-02-02
淺談緩沖字符流 BufferedReader BufferedWriter用法
這篇文章主要介紹了緩沖字符流 BufferedReader BufferedWriter的用法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
springboot實現(xiàn)跨域的五種方式總結(jié)
在Spring Boot中實現(xiàn)跨域,可以采用全局跨域和局部跨域兩種方式,下面這篇文章主要給大家介紹了關于springboot實現(xiàn)跨域的五種方式,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-01-01
java?Map接口子類HashMap遍歷與LinkedHashMap詳解
這篇文章主要介紹了java?Map接口子類HashMap遍歷與LinkedHashMap詳解,Map接口下的集合與Collection接口下的集合,它們存儲數(shù)據(jù)的形式不同,感興趣的小伙伴可以參考下面文章詳細內(nèi)容介紹2022-06-06

