自己动手实现一次性图片验证码

  以前碰到注册登录需要图形验证码的时候,一般都是到网络上直接淘一个,也不读代码,直接就用了,今天静下来,从头到尾读了一遍代码,自己又照着写了一遍,加上了完整的注释,下面介绍步骤,首先新建一个web项目,src新建一个class类:

  1 import java.awt.BasicStroke;
  2 import java.awt.Color;
  3 import java.awt.Font;
  4 import java.awt.Graphics;
  5 import java.awt.Graphics2D;
  6 import java.awt.image.BufferedImage;
  7 import java.io.FileNotFoundException;
  8 import java.io.FileOutputStream;
  9 import java.io.IOException;
 10 import java.io.OutputStream;
 11 import java.util.Random;
 12 
 13 import javax.imageio.ImageIO;
 14 
 15 public class VCode {
 16 
 17     private int w;// 图片宽
 18     private int h;// 图片高
 19     private Color bgColor = new Color(240, 240, 240);// 背景色
 20     private Random random = new Random();// 随机数对象
 21     // 设置字体范围
 22     private String[] fontNames = { "宋体", "华文楷体", "黑体", "华文新魏", "华文隶书", "微软雅黑",
 23             "楷体" };
 24     //设置字体样式范围
 25     private int[] fontstyles = { 0, 1, 2, 3 };
 26     //设置字号范围
 27     private int[] fontSizes={24,25,26,27,28};
 28     //设置所有字符串范围
 29     private String codes="abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
 30     
 31     
 32     //无参构造方法
 33     public VCode() {
 34     }
 35     
 36     //带宽和高的构造函数 
 37     public VCode(int w, int h) {
 38         super();
 39         this.w = w;
 40         this.h = h;
 41     }
 42 
 43 
 44     // 返回一张背景图片
 45     private BufferedImage createImage() {
 46         /**
 47          * 1:创建图片 2:设置背景色
 48          */
 49         // 1:创建图片
 50         BufferedImage img = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
 51         // 2:设置背景色
 52         Graphics g = img.getGraphics();
 53         g.setColor(bgColor);
 54         g.fillRect(0, 0, w, h);
 55 
 56         return img;
 57     }
 58 
 59     // 随机返回字体颜色
 60     private Color randomColor() {
 61         int r = random.nextInt(256);
 62         int g = random.nextInt(256);
 63         int b = random.nextInt(256);
 64         return new Color(r, g, b);
 65     }
 66 
 67     // 随机返回字体样式
 68     private Font randomFont() {
 69         //随机生成字体下标,随机从给定的范围内获取一个字体
 70         int index=random.nextInt(fontNames.length);
 71         String name=fontNames[index];
 72 
 73         //随机生成字体样式下表,随机从给定的返回内获取到一个字体样式
 74         index=random.nextInt(fontstyles.length);
 75         int style = fontstyles[index];
 76 
 77         //随机生成字体大小下标,随机从给定的返回内获取到一个字体大小
 78         index=random.nextInt(fontSizes.length);
 79         int size = fontSizes[index];
 80 
 81         return new Font(name, style, size);
 82     }
 83 
 84     // 随机返回字体内容
 85     private String randomChar() {
 86         int index=random.nextInt(codes.length());
 87         
 88         return codes.charAt(index)+"";
 89     }
 90 
 91     //随即返回几条干扰线
 92     private void getLine(BufferedImage img){
 93         //设置干扰线的宽度为1.5倍宽,随机画五条
 94         Graphics2D g=(Graphics2D)img.getGraphics();
 95         g.setColor(Color.BLACK);
 96         g.setStroke(new BasicStroke(1.5f));
 97         for(int i=0;i<5;i++){
 98             int x1=random.nextInt(w);
 99             int y1=random.nextInt(h);
100             int x2=random.nextInt(w);
101             int y2=random.nextInt(h);
102             g.drawLine(x1, y1, x2, y2);
103             
104         }
105         
106     }
107     // 用户调用该方法获取图片
108     public BufferedImage getImage() {
109         /**
110          * 随机生成字符,字符范围0-9A-Za-z, 设置字体,字号,是否粗体 都是随机 字符的颜色
111          */
112         BufferedImage img = createImage();
113 
114         this.getLine(img);
115         // 获取画笔
116         Graphics g = img.getGraphics();
117         // 画内容
118         for (int i = 0; i < 4; i++) {
119             g.setColor(this.randomColor());// 获取随机颜色
120             g.setFont(this.randomFont());// 获取随机字体
121             g.drawString(this.randomChar(), w / 4 * i, h - 5);// 获取字符串随机内容
122         }
123         return img;
124     }
125     
126     //用户调用该方法保存图片到本地
127     public void saveImage(BufferedImage img,OutputStream ous){
128         
129         try {
130             ImageIO.write(img, "JPEG", ous);
131         } catch (FileNotFoundException e) {
132             e.printStackTrace();
133         } catch (IOException e) {
134             e.printStackTrace();
135         }
136     }
137 }

接下来新建一个servlet:

 1 package com.wang.verifySode;
 2 
 3 import java.awt.Color;
 4 import java.awt.Font;
 5 import java.awt.Graphics;
 6 import java.awt.image.BufferedImage;
 7 import java.io.FileOutputStream;
 8 import java.io.IOException;
 9 import java.io.PrintWriter;
10 
11 import javax.imageio.ImageIO;
12 import javax.servlet.ServletException;
13 import javax.servlet.http.HttpServlet;
14 import javax.servlet.http.HttpServletRequest;
15 import javax.servlet.http.HttpServletResponse;
16 
17 public class BServlet extends HttpServlet {
18 
19     public void doGet(HttpServletRequest request, HttpServletResponse response)
20             throws ServletException, IOException {
21         
22         VCode v=new VCode(70, 35);
23         BufferedImage img=v.getImage();
24         v.saveImage(img, response.getOutputStream());
25         
26     }
27 
28 }

在你需要加验证码的地方,如注册页面中适当地方加入:

<img id="img" alt="" src="/tools/BServlet"><a href="javascript:changeNext()">看不清楚,换一张</a>

这里需要说一下,因为验证码有一个"看不清楚 换一张"的功能,所以需要加一个javascript函数,如下:

<script type="text/javascript">
    function changeNext(){
        var a=document.getElementById("img");
        a.src="/tools/BServlet?a="+new Date().getTime();
    }
</script>

为了避免浏览器的图片缓存问题而导致点击后无法随机下一张图片,这里在超链接后面加一个永不重复的参数,可以避免这个问题,见上面代码.

配置项目,开浏览器,运行,完美~

 

posted @ 2016-01-23 00:08  冬至饮雪  阅读(1247)  评论(0编辑  收藏  举报