在一个门户网站中用户登陆时最好设置图片验证码,防止非法用户的机器暴力攻击,在页面通过创建一个图形文字或者汉字提供用户输入。这就防止机器的暴力攻击行为。常用的方式使用AWT创建一个图片添加一些随机产生若干条干扰线,使图象中的认证码不易被其它程序探测到。同时写入Web页面,在用户请求时随机生成。
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;
import javax.imageio.ImageIO;
/**
* 图片验证码工具类
*
* @version 创建时间2007-11-1 上午11:28:14 类说明 帮助生成图片验证码
*/
public class ColorplateUtil {
/**
* 随机数类型 数字类型
*/
public static final int NUMBER_TYPE = 0;
/**
* 随机数类型 字母类型
*/
public static final int LETTER_TYPE = 1;
/**
* 随机数类型 中文类型
*/
public static final int CHINESE_TYPE = 2;
/**
* 随机数类型 数字字母类型
*/
public static final int NUMBER_LETTER_TYPE = 3;
/**
* 随机数类型 中文字母类型
*/
public static final int CHINESE_LETTER_TYPE = 4;
/**
* 随机数类型 中文数字类型
*/
public static final int CHINESE_NUMBER_TYPE = 5;
/**
* 随机数类型 混合类型
*/
public static final int ALL_TYPE = 6;
/**
* 干扰线类型 线型
*/
public static final int DISTURB_LINE = 0;
/**
* 干扰线类型 平面
*/
public static final int DISTURB_PANEL = 2;
/**
* 图片高度
*/
private int height;
/**
* 图片宽度
*/
private int width;
/**
* 图片字体
*/
private Font font;
/**
* 干扰线数量
*/
private int disturb;
/**
* 是否加上边框
*/
private boolean rim = false;
/**
* 随机类型(数字,汉字,字母,数字字母,汉字字母,汉字数字,混合) 默认为[数字类型]
*/
private int contentType = 0;
/**
* 随机数数量(默认为4)
*/
private int randCount = 4;
/**
* 干扰线的类型
*/
private int disturbType = 0;
/**
* 是否为彩色验证码
*/
private boolean color = true;
/**
* 绘制验证码
*
* @param request
* @param response
* @throws IOException
*/
public void paint(HttpServletRequest request, HttpServletResponse response)
throws IOException {
response.setContentType("image/jpeg");
response.setHeader("Pragma", "No-cache");
response.setHeader("Cache-Control", "no-cache");
response.setDateHeader("Expires", 0);
HttpSession session = request.getSession();
// 初始化参数
this.init();
// 在内存中创建图象
BufferedImage image = new BufferedImage(width, height,
BufferedImage.TYPE_INT_RGB);
// 获取图形上下文
Graphics graphics = image.getGraphics();
// 生成随机类
Random random = new Random();
// 设定背景色(将此图形上下文的当前颜色设置为指定颜色。使用此图形上下文的所有后续图形操作均使用这个指定的颜色)
if (this.isColor()) {
graphics.setColor(getRandColor(200, 250));
} else {
graphics.setColor(new Color(0xDCDCDC));
}
// 填充指定的矩形,使用图形上下文的当前颜色填充该矩形
graphics.fillRect(0, 0, this.getWidth(), this.getHeight());
// 设定字体
graphics.setFont(this.getFont());
if (this.isRim()) {
// 画边框
graphics.setColor(Color.black);
graphics.drawRect(0, 0, this.getWidth() - 1, this.getHeight() - 1);
}
// 制造干扰线
this.markDisturb(graphics);
// 备选字体
String[] fontTypes = { "\u5b8b\u4f53", "\u65b0\u5b8b\u4f53",
"\u9ed1\u4f53", "\u6977\u4f53", "\u96b6\u4e66" };
// 备选字体数量
int fontTypesLength = fontTypes.length;
// 取随机产生的认证码(参数count位数字)
String sRand = "";
for (int i = 0; i < this.getRandCount(); i++) {
String rand = this.getRandString();
sRand += rand;
// 将认证码显示到图象中
// 调用函数出来的颜色相同,可能是因为种子太接近,所以只能直接生成
if (this.isColor()) {
graphics.setColor(new Color(20 + random.nextInt(110),
20 + random.nextInt(110), 20 + random.nextInt(110)));
} else {
graphics.setColor(Color.black);
}
graphics.setFont(new Font(
fontTypes[random.nextInt(fontTypesLength)], Font.BOLD,
18 + random.nextInt(6)));
if (this.getContentType() == ColorplateUtil.ALL_TYPE
|| this.getContentType() == ColorplateUtil.CHINESE_LETTER_TYPE
|| this.getContentType() == ColorplateUtil.CHINESE_NUMBER_TYPE
|| this.getContentType() == ColorplateUtil.CHINESE_TYPE) {
// 将此文字画到图片上
graphics.drawString(rand, 20 * i + 10 + random.nextInt(8), 20);
} else {
// 将文字画到图片上
graphics.drawString(rand, 13 * i + 6, 16);
}
}
// 将认证码存入session
session.setAttribute("rand", sRand);
// 图象生效
graphics.dispose();
ServletOutputStream responseOutputStream = response.getOutputStream();
// 输出图象到页面
ImageIO.write(image, "JPEG", responseOutputStream);
// 以下关闭输入流!
responseOutputStream.flush();
responseOutputStream.close();
}
/**
* 制造干扰线
*
* @param graphics
*/
private void markDisturb(Graphics graphics) {
Random random = new Random();
// 如果是线形干扰点
if (this.getDisturbType() == ColorplateUtil.DISTURB_LINE) {
// 随机产生若干条干扰线,使图象中的认证码不易被其它程序探测到
graphics.setColor(getRandColor(160, 200));
for (int i = 0; i < this.getDisturb(); i++) {
int x = random.nextInt(this.getWidth());
int y = random.nextInt(this.getHeight());
int _x = random.nextInt(12);
int _y = random.nextInt(12);
graphics.drawLine(x, y, x + _x, y + _y);
}
} else if (this.getDisturbType() == ColorplateUtil.DISTURB_PANEL) {
for (int i = 0; i < 20; i++) {
int x = random.nextInt(this.getWidth());
int y = random.nextInt(this.getHeight());
int _x = random.nextInt(12);
int _y = random.nextInt(12);
graphics.drawOval(x, y, x + _x, y + _y);
}
}
}
/**
* 产生随机数据
*
* @return
*/
private String getRandString() {
Random random = new Random();
String result = "";
// 备选数据 数字
String number = "0123456789";
// 备选数据 字母
String letter = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
// 备选数据 汉字,剔除一些不雅的汉字
String chinese = "\u7684\u4e00\u4e86\u662f\u6211\u4e0d\u5728\u4eba\u4eec\u6709\u6765\u4ed6\u8fd9\u4e0a\u7740\u4e2a\u5730\u5230\u5927\u91cc\u8bf4\u5c31\u53bb\u5b50\u5f97\u4e5f\u548c\u90a3\u8981\u4e0b\u770b\u5929\u65f6\u8fc7\u51fa\u5c0f\u4e48\u8d77\u4f60\u90fd\u628a\u597d\u8fd8\u591a\u6ca1\u4e3a\u53c8\u53ef\u5bb6\u5b66\u53ea\u4ee5\u4e3b\u4f1a\u6837\u5e74\u60f3\u751f\u540c\u8001\u4e2d\u5341\u4ece\u81ea\u9762\u524d\u5934\u9053\u5b83\u540e\u7136\u8d70\u5f88\u50cf\u89c1\u4e24\u7528\u5979\u56fd\u52a8\u8fdb\u6210\u56de\u4ec0\u8fb9\u4f5c\u5bf9\u5f00\u800c\u5df1\u4e9b\u73b0\u5c71\u6c11\u5019\u7ecf\u53d1\u5de5\u5411\u4e8b\u547d\u7ed9\u957f\u6c34\u51e0\u4e49\u4e09\u58f0\u4e8e\u9ad8\u624b\u77e5\u7406\u773c\u5fd7\u70b9\u5fc3\u6218\u4e8c\u95ee\u4f46\u8eab\u65b9\u5b9e\u5403\u505a\u53eb\u5f53\u4f4f\u542c\u9769\u6253\u5462\u771f\u5168\u624d\u56db\u5df2\u6240\u654c\u4e4b\u6700\u5149\u4ea7\u60c5\u8def\u5206\u603b\u6761\u767d\u8bdd\u4e1c\u5e2d\u6b21\u4eb2\u5982\u88ab\u82b1\u53e3\u653e\u513f\u5e38\u6c14\u4e94\u7b2c\u4f7f\u5199\u519b\u5427\u6587\u8fd0\u518d\u679c\u600e\u5b9a\u8bb8\u5feb\u660e\u884c\u56e0\u522b\u98de\u5916\u6811\u7269\u6d3b\u90e8\u95e8\u65e0\u5f80\u8239\u671b\u65b0\u5e26\u961f\u5148\u529b\u5b8c\u5374\u7ad9\u4ee3\u5458\u673a\u66f4\u4e5d\u60a8\u6bcf\u98ce\u7ea7\u8ddf\u7b11\u554a\u5b69\u4e07\u5c11\u76f4\u610f\u591c\u6bd4\u9636\u8fde\u8f66\u91cd\u4fbf\u6597\u9a6c\u54ea\u5316\u592a\u6307\u53d8\u793e\u4f3c\u58eb\u8005\u5e72\u77f3\u6ee1\u65e5\u51b3\u767e\u539f\u62ff\u7fa4\u7a76\u5404\u516d\u672c\u601d\u89e3\u7acb\u6cb3\u6751\u516b\u96be\u65e9\u8bba\u5417\u6839\u5171\u8ba9\u76f8\u7814\u4eca\u5176\u4e66\u5750\u63a5\u5e94\u5173\u4fe1\u89c9\u6b65\u53cd\u5904\u8bb0\u5c06\u5343\u627e\u4e89\u9886\u6216\u5e08\u7ed3\u5757\u8dd1\u8c01\u8349\u8d8a\u5b57\u52a0\u811a\u7d27\u7231\u7b49\u4e60\u9635\u6015\u6708\u9752\u534a\u706b\u6cd5\u9898\u5efa\u8d76\u4f4d\u5531\u6d77\u4e03\u5973\u4efb\u4ef6\u611f\u51c6\u5f20\u56e2\u5c4b\u79bb\u8272\u8138\u7247\u79d1\u5012\u775b\u5229\u4e16\u521a\u4e14\u7531\u9001\u5207\u661f\u5bfc\u665a\u8868\u591f\u6574\u8ba4\u54cd\u96ea\u6d41\u672a\u573a\u8be5\u5e76\u5e95\u6df1\u523b\u5e73\u4f1f\u5fd9\u63d0\u786e\u8fd1\u4eae\u8f7b\u8bb2\u519c\u53e4\u9ed1\u544a\u754c\u62c9\u540d\u5440\u571f\u6e05\u9633\u7167\u529e\u53f2\u6539\u5386\u8f6c\u753b\u9020\u5634\u6b64\u6cbb\u5317\u5fc5\u670d\u96e8\u7a7f\u5185\u8bc6\u9a8c\u4f20\u4e1a\u83dc\u722c\u7761\u5174\u5f62\u91cf\u54b1\u89c2\u82e6\u4f53\u4f17\u901a\u51b2\u5408\u7834\u53cb\u5ea6\u672f\u996d\u516c\u65c1\u623f\u6781\u5357\u67aa\u8bfb\u6c99\u5c81\u7ebf\u91ce\u575a\u7a7a\u6536\u7b97\u81f3\u653f\u57ce\u52b3\u843d\u94b1\u7279\u56f4\u5f1f\u80dc\u6559\u70ed\u5c55\u5305\u6b4c\u7c7b\u6e10\u5f3a\u6570\u4e61\u547c\u6027\u97f3\u7b54\u54e5\u9645\u65e7\u795e\u5ea7\u7ae0\u5e2e\u5566\u53d7\u7cfb\u4ee4\u8df3\u975e\u4f55\u725b\u53d6\u5165\u5cb8\u6562\u6389\u5ffd\u79cd\u88c5\u9876\u6025\u6797\u505c\u606f\u53e5\u533a\u8863\u822c\u62a5\u53f6\u538b\u6162\u53d4\u80cc\u7ec6";
// 备选字体
String[] fontTypes = { "\u5b8b\u4f53", "\u65b0\u5b8b\u4f53",
"\u9ed1\u4f53", "\u6977\u4f53", "\u96b6\u4e66" };
int fontTypesLength = fontTypes.length;
// 如果是数字类型
if (this.getContentType() == ColorplateUtil.NUMBER_TYPE) {
int start = random.nextInt(number.length());
result = number.substring(start, start + 1);
} else if (this.getContentType() == ColorplateUtil.LETTER_TYPE) {// 如果是字母类型
int start = random.nextInt(letter.length());
result = letter.substring(start, start + 1);
} else if (this.getContentType() == ColorplateUtil.CHINESE_TYPE) {// 如果是中文类型
int start = random.nextInt(chinese.length());
result = chinese.substring(start, start + 1);
} else if (this.getContentType() == ColorplateUtil.NUMBER_LETTER_TYPE) {// 如果是数字字母类型
int start = random.nextInt((number + letter).length());
result = (number + letter).substring(start, start + 1);
} else if (this.getContentType() == ColorplateUtil.CHINESE_LETTER_TYPE) {// 如果是中文字母类型
int start = random.nextInt((chinese + letter).length());
result = (chinese + letter).substring(start, start + 1);
} else if (this.getContentType() == ColorplateUtil.CHINESE_NUMBER_TYPE) {// 如果是中文数字类型
int start = random.nextInt((chinese + number).length());
result = (chinese + number).substring(start, start + 1);
} else {// 如果是混合类型
int start = random.nextInt((chinese + number + letter).length());
result = (chinese + number + letter).substring(start, start + 1);
}
return result;
}
/**
* 初始化本类
*
*/
private void init() {
// 如果没有设置高度设置成默认高度
if (this.height <= 15) {
this.height = 30;
}
// 如果没有设置宽度设置成默认宽度
if (this.width <= 50) {
this.width = 120;
}
// 如果没有设置字体,设置为默认字体
if (this.font == null) {
font = new Font("Times New Roman", Font.PLAIN, 18);
}
// 如果没有设置干扰线的数量,设置默认干扰线的数量
if (this.disturb < 50) {
this.disturb = 200;
}
// 如果没有设置随机文字类型,设置默认随机文字类型
if (this.contentType > 6 || this.contentType < 0) {
this.contentType = 0;
}
// 如果没有设置随机数量,设置默认随机数量
if (this.randCount < 0) {
this.randCount = 4;
}
// 如果没有设置干扰线类型,设置默认干扰线类型
if (this.disturbType < 0) {
this.disturbType = 0;
}
}
/**
* 给定范围获得随机颜色
*
* @param fc
* @param bc
* @return
*/
private Color getRandColor(int fc, int bc) {//
Random random = new Random();
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);
}
/**
* 获取图片的高度
*
* @return
*/
public int getHeight() {
return height;
}
/**
* 设置图片的高度
*
* @param height
*/
public void setHeight(int height) {
this.height = height;
}
/**
* 获取图片的宽度
*
* @return
*/
public int getWidth() {
return width;
}
/**
* 设置图片的宽度
*
* @param width
*/
public void setWidth(int width) {
this.width = width;
}
/**
* 获取图片上文字的字体
*
* @return
*/
public Font getFont() {
return font;
}
/**
* 设置图片上文字的字体
*
* @param font
*/
public void setFont(Font font) {
this.font = font;
}
/**
* 获取随机数据的类型
*
* @return
*/
public int getContentType() {
return contentType;
}
/**
* 设置随机数据的类型
*
* @param contentType
*/
public void setContentType(int contentType) {
this.contentType = contentType;
}
/**
* 获取干扰线的数量
*
* @return
*/
public int getDisturb() {
return disturb;
}
/**
* 设置干扰线的数量
*
* @param disturb
*/
public void setDisturb(int disturb) {
this.disturb = disturb;
}
/**
* 获取随机数据的个数
*
* @return
*/
public int getRandCount() {
return randCount;
}
/**
* 设置随机数据的个数
*
* @param randCount
*/
public void setRandCount(int randCount) {
this.randCount = randCount;
}
/**
* 判断是否给图片加上边框
*
* @return
*/
public boolean isRim() {
return rim;
}
/**
* 设置是否给图片加上边框
*
* @param rim
*/
public void setRim(boolean rim) {
this.rim = rim;
}
/**
* 获取干扰线的类型
*
* @return
*/
public int getDisturbType() {
return disturbType;
}
/**
* 设置干扰线的类型
*
* @param disturbType
*/
public void setDisturbType(int disturbType) {
this.disturbType = disturbType;
}
/**
* 判断是否为彩色验证码
*
* @return
*/
public boolean isColor() {
return color;
}
/**
* 设置为是否为彩色验证码
*
* @param color
*/
public void setColor(boolean color) {
this.color = color;
}
}
相关推荐
Java图片验证码工具类是一种在Web应用中用于防止恶意自动化操作(如机器人)的常用安全机制。它通过生成随机的字符序列并将其显示为图片来工作,用户需要输入看到的字符才能完成验证。以下是对这个Java工具类的详细...
在实际应用中,验证码工具类通常会封装这些功能,提供简单的API供其他部分代码调用。例如,可能有一个`generate()`方法用于生成验证码,一个`getImageBytes()`方法用于获取验证码图像的字节数组,以及一个`validate...
示例代码可以帮助理解如何在实际项目中整合和使用验证码工具类,包括如何调用、如何配置以及验证逻辑的实现。 综上所述,这两个版本的验证码工具类提供了一种简单而有效的方法来增强Web应用的安全性。它们利用了...
本项目提供的"验证码工具类.rar"是一个Java实现的简单验证码生成器,能够生成四位数字的随机验证码,并具有字体和颜色的随机性,以及添加干扰线以增加机器识别的难度。 首先,我们来看核心类`VerifyCode.java`。这...
调用类自动生成验证码,可用于web项目的注册登录。详情看说明
验证码在IT领域,尤其是Web...总的来说,PHP验证码工具类如KCAPTCHA是Web开发者防止恶意自动化攻击的重要武器。合理使用并自定义这类工具,可以有效地平衡用户体验与网站安全,是构建健壮Web应用程序不可或缺的一部分。
描述中同样强调了"java生成图片验证码",这意味着我们将专注于Java实现的具体步骤和技术。 标签"java servlet"表明我们将在Servlet环境下进行操作。Servlet是Java EE(现在称为Jakarta EE)的一部分,主要用于开发...
用servlet编写JAVA web的验证码类,用户绘制网页上的验证码提示信息!
在Web开发中,工具类是不可或缺的部分,它们封装了常用且重复的功能,提高了代码的可重用性和效率。以下是对标题和描述中提及的几个关键知识点的详细解释: 1. **正则校验**: 正则表达式(Regular Expression)是...
Java图形验证码生成工具类是用于在Web页面上创建安全的随机验证码图像的程序。验证码的主要目的是防止自动机器人或恶意软件进行非法操作,如批量注册、恶意登录等。它通过要求用户输入显示在图像中的随机字符序列来...
在IT领域,验证码是一种常见的安全机制,用于防止自动化的机器人或恶意软件进行非法操作。...这种验证码机制结合了语言和数学,既增加了安全性,又保持了用户体验的良好,是Web应用中防止恶意自动化操作的有效手段。
总而言之,验证码工具类是一种重要的Web安全技术,它在防止自动化攻击、提高用户操作安全性方面发挥着关键作用。对于想要在Web项目中实现验证码功能的开发人员来说,本文所提供的VeriyCodeUtils类是一个不错的参考。...
验证码判断工具类CpachaUtil.java
总的来说,Java提供了一套强大的图像处理库,使得我们可以轻松地创建出功能完善的图片验证码工具类。通过学习和理解`VerifyImage`类的实现,开发者能够更好地理解和掌握Java在安全领域中的应用,同时提升应用的安全...
8. **工具**:标签还提到了“工具”,可能是指该验证码系统可以作为开发者的工具,快速集成到自己的Web应用中,提高开发效率。 综上所述,JAVA EasyCode验证码系统是一个集成了JAVA编程、图像处理、随机数生成、WEB...
验证码生成类大全是一种用于网络安全验证的重要工具,它主要用于防止恶意自动化程序(如机器人)进行非法操作,例如批量注册、刷票或数据抓取。在Web应用中,验证码通常需要用户在提交表单前输入图像上显示的一串...
综上所述,Java验证码工具类的实现涉及到图像处理、随机数生成、Web交互等多个方面,是保障Web应用安全性的重要组成部分。通过合理的实现和不断优化,可以有效防止自动化攻击,保护用户的账户安全。
在这个Java开发的验证码工具类中,它提供了生成和验证验证码的功能,同时包含了文档说明和Spring MVC框架的集成示例。 验证码(CAPTCHA)的基本原理是生成一段随机的字符串或图像,用户需要正确输入才能完成相应的...
在Web开发中,验证码输入功能是一项重要的安全机制,它用于防止自动化的机器人或者恶意脚本进行非法操作,比如批量注册、频繁登录等。验证码的主要作用是确保用户是真实的人,而不是计算机程序。在这个主题中,我们...
7. **显示验证码**:在Web应用中,这个生成的验证码图像会被显示在网页上供用户输入验证。 8. **处理用户输入**:用户输入的验证码需要与服务器端保存的原始验证码进行比对,验证成功后才允许执行相应操作。 总的...