`
longgangbai
  • 浏览: 7254026 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Web中图片验证码工具类

阅读更多

     在一个门户网站中用户登陆时最好设置图片验证码,防止非法用户的机器暴力攻击,在页面通过创建一个图形文字或者汉字提供用户输入。这就防止机器的暴力攻击行为。常用的方式使用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;
 }
}

分享到:
评论

相关推荐

    verifycode生成验证码工具类

    调用类自动生成验证码,可用于web项目的注册登录。详情看说明

    验证码工具类.zip

    web开发验证码工具类 servlet版和struts2(Action)版两个文件夹,资源包含验证码源码.java文件,.jar包,和使用说明,以及代码样例

    验证码工具类

    用servlet编写JAVA web的验证码类,用户绘制网页上的验证码提示信息!

    java图形验证码生成工具类 web页面校验验证码

    主要为大家详细介绍了java图形验证码生成工具类,web页面校验验证码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

    验证码判断工具类CpachaUtil.java

    验证码判断工具类CpachaUtil.java

    web项目里用到的工具类(正则校验、图片验证码、json解析、订单号生成)

    包括web项目里面非常常用的重复性代码,如校验用户名、手机号、身份证号、密码、等等,是否符合安全规则。 生成绝不重复的订单号。 生成图片验证码。 json解析。 全是自己手写,供大家复用。

    验证码生成工具类+验证码验证学习笔记

    验证码生成工具类参考自:java图形验证码生成工具类及web页面校验验证码,万分感谢! 效果图: 代码如下: package com.my.demo.session.others; import javax.imageio.ImageIO; import java.awt.*; import java....

    java utils 工具类

    java 工具类,包含上传下载工具类,数据库工具类,验证码工具类,分页工具类,字符编码过滤工具类,字符串工具类 提醒:我已打包,在/WEB-INF/lib/itbofeng-utils.jar中,可以看到源码。 测试下载文件时修改servlet...

    java写的几种方便web开发工具类源码

    工具类里面有关于java bean 转map map转Javabean 可以支持日期类型转换。关于发邮件可见附件工具类,验证码工具类 编码过滤工具类 加密解密工具类,用c3po连接数据库的

    .Net 50多种工具类

    个人收集的c#和.net的工具类 包含FTP、邮件、pdf、csv、xml、图片、图形、条形码、序列化、压缩解压、分词、汉字转拼音、时间操作、万能数据转换、网络、文件操作、弹窗、正则、字符串操作、web验证码、水印缩略图、...

    随机验证码,支持数字,字母混合,包含验证码校验等。

    这是一个支持数字,字母混合的随机验证码工具类,包含验证码校验等。

    kaptcha验证码

    非常轻巧,而且配置部署相当简单的一种验证码,只需要引入jar包,并在web.xml中配置一下即可使用。

    SpringBoot+Shiro登录验证码制作

    生成验证码的工具类的写法: import javax.imageio.ImageIO; import java.awt.*; import java.awt.image.BufferedImage; import java.io.*; import java.util.Random; /** * @author lrx * @description: TODO ...

    通过servlet生成验证码

    通过servlet生成验证码的小工具类,使用时在web.xml里面配置,页面上引用时利用img标签引用这个servlet就可以实现验证码实例: &lt;img src="checkCode.do" id="img" height="30px"&gt;&lt;/a&gt; 实现点击图片时更换验证码

    软件测试工具中QTP处理验证码的一种方法

    通过识别图片中的文字来获取验证码, 软件测试工具中QTP处理验证码的一种方法QTP在测试WEB站点应用程序时,通常会碰到验证码的问题(尤其是论坛类的站点),对于这类问题,通常的做法是利用OCR...

    单击局部刷新的验证码控件

    2、在Web.config中的system.web节点间添加即可以使用! *" type="ValidateImage.ValidateImageHandler" /&gt; &lt;/httpHandlers&gt; 3、设置属性请将属性设置为按分类排序,即可看到验证码控件的属性。 4、保留所有...

    软件测试工具QTP处理验证码的一种方法

    通过识别图片中的文字来获取验证码,但是软件测试工具QTP处理验证码的一种方法 QTP在测试WEB站点应用程序时,通常会碰到验证码的问题(尤其是论坛类的站点),对于这类问题,通常的做法是利用OCR...

    .Net的jsp里添加验证码文件

    这个压缩包里面有说明怎么使用验证码代码的功能,只需要添加几个类后,再在html页面修改一些就行!

    vcg(验证码控件)

    在Default.aspx中新增Web控件TextBox:textCode,作为验证码输入;在Default.aspx中新增Web控件Lable:lblMessage,作为验证结果输出; f) 在Default.aspx.cs中添加验证函数: protected void btnValidate_...

Global site tag (gtag.js) - Google Analytics