zoukankan      html  css  js  c++  java
  • java web 验证码生成

    先看看效果图:

    一般需求应该够用了,有线条干扰线,斑点,字体小幅度的旋转。。

    还没有与session挂钩上,所以只能用于简单的测试用例。

    直接上代码,看代码注释。。。

      1 package com.act262.demo;
      2 
      3 import java.awt.Color;
      4 import java.awt.Font;
      5 import java.awt.Graphics;
      6 import java.awt.Graphics2D;
      7 import java.awt.image.BufferedImage;
      8 import java.io.IOException;
      9 import java.io.OutputStream;
     10 import java.util.Random;
     11 
     12 import javax.imageio.ImageIO;
     13 import javax.servlet.ServletException;
     14 import javax.servlet.annotation.WebServlet;
     15 import javax.servlet.http.HttpServlet;
     16 import javax.servlet.http.HttpServletRequest;
     17 import javax.servlet.http.HttpServletResponse;
     18 
     19 import sun.java2d.loops.DrawLine;
     20 
     21 /**
     22  * 输出随机的验证码
     23  */
     24 @WebServlet({ "/CheckCode", "/checkCode.do" })
     25 public class CheckCodeServlet extends HttpServlet {
     26     private static final long serialVersionUID = 1L;
     27 
     28     /* 宽度 */
     29     private final int WIDTH = 100;
     30     /* 高度 */
     31     private final int HEIGHT = 20;
     32     /* 生成验证码的个数 */
     33     private final int COUNT = 4;
     34     /* 干扰线条数 */
     35     private final int LINE_ROW = 6;
     36 
     37     /* 输出的基本码表,如果使用中文,则使用utf-8的码表,类似 ue234 ,而且应该使用常用字,避免出现偏僻字 */
     38     private final char[] BASECODE = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h',
     39             'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u',
     40             'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H',
     41             'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U',
     42             'V', 'W', 'X', 'Y', 'Z', '0', '1', '2', '3', '4', '5', '6', '7',
     43             '8', '9' };
     44     /* 随机数发生器 */
     45     private Random random;
     46     private BufferedImage image;
     47 
     48     // 写出数据
     49     private void write(OutputStream output) throws IOException {
     50         random = new Random();
     51 
     52         image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_RGB);
     53         Graphics graphics = image.getGraphics();
     54 
     55         setBackground(graphics);
     56         drawBorder(graphics);
     57         drawDot(graphics);
     58         drawLine(graphics);
     59         drawString(graphics);
     60 
     61         ImageIO.write(image, "jpg", output);
     62     }
     63 
     64     //写字
     65     private void drawString(Graphics graphics) {
     66         StringBuffer sb = new StringBuffer();
     67         Font font = new Font("宋体", Font.BOLD, 18);
     68         graphics.setFont(font);
     69         graphics.setColor(Color.BLACK);
     70 
     71         for (int i = 0; i < COUNT; i++) {
     72             String ch = String
     73                     .valueOf(BASECODE[random.nextInt(BASECODE.length)]);
     74             sb.append(ch);
     75 
     76             // 设置位置
     77             int x = i * 20 + random.nextInt(12) + 10;
     78             int y = random.nextInt(HEIGHT / 3) + 12;
     79 
     80             // 旋转字体
     81             double theta = Math.PI / 180 * random.nextInt(20);
     82             // rotate(graphics, theta);
     83 
     84             graphics.drawString(ch, x, y);
     85 
     86             // 恢复。。
     87             // rotate(graphics, -theta);
     88         }
     89 
     90         System.out.println("验证码:" + sb.toString());
     91     }
     92 
     93     //旋转
     94     private void rotate(Graphics graphics, double theta) {
     95         ((Graphics2D) graphics).rotate(theta);
     96     }
     97 
     98     // 画随机线条
     99     private void drawLine(Graphics graphics) {
    100         for (int i = 0; i < LINE_ROW; i++) {
    101             int x1 = random.nextInt(WIDTH);
    102             int y1 = random.nextInt(HEIGHT);
    103             int x2 = random.nextInt(WIDTH);
    104             int y2 = random.nextInt(HEIGHT);
    105             setRandomColor(graphics);
    106             graphics.drawLine(x1, y1, x2, y2);
    107         }
    108     }
    109 
    110     // 画斑点
    111     private void drawDot(Graphics graphics) {
    112         graphics.setColor(Color.red);
    113         for (int i = 0; i < WIDTH; i++) {
    114             int x = i;
    115             int y = random.nextInt(HEIGHT);
    116             int r = random.nextInt(2);
    117             // graphics.fillOval(x, y, r, r);
    118             graphics.drawOval(x, y, r, r);
    119         }
    120     }
    121 
    122     // 画边框
    123     private void drawBorder(Graphics graphics) {
    124         graphics.setColor(Color.BLACK);
    125         graphics.drawRect(1, 1, WIDTH - 2, HEIGHT - 2);
    126     }
    127 
    128     // 设置背景
    129     private void setBackground(Graphics graphics) {
    130         graphics.setColor(Color.WHITE);
    131         graphics.fillRect(0, 0, WIDTH, HEIGHT);// 填充背景色
    132     }
    133 
    134     // 设置随机的画笔颜色
    135     private void setRandomColor(Graphics g) {
    136         g.setColor(new Color(random.nextInt(255), random.nextInt(255), random
    137                 .nextInt(255)));
    138     }
    139 
    140     protected void doGet(HttpServletRequest request,
    141             HttpServletResponse response) throws ServletException, IOException {
    142 
    143         response.setContentType("image/jpeg");
    144         response.setHeader("Expires", "-1");
    145         response.setHeader("Cache-Control", "no-cache");
    146 
    147         // 写出数据
    148         write(response.getOutputStream());
    149 
    150     }
    151 
    152     protected void doPost(HttpServletRequest request,
    153             HttpServletResponse response) throws ServletException, IOException {
    154         // TODO Auto-generated method stub
    155     }
    156 
    157 }

    不足的是,这个验证码数量与字体大小还有图片的大小之间的关系比较难搞,我是一点一点计算调试出来的,有些时候图片上的字体会跑位,这就是画图的难点了。如果修改了数量,字体大小,这些位置关系得重新计算才行。

    后续修改

    还是這个验证码

      1 package com.act262.demo;
      2 
      3 import java.awt.Color;
      4 import java.awt.Font;
      5 import java.awt.Graphics;
      6 import java.awt.Graphics2D;
      7 import java.awt.image.BufferedImage;
      8 import java.io.IOException;
      9 import java.io.OutputStream;
     10 import java.util.Random;
     11 
     12 import javax.imageio.ImageIO;
     13 import javax.servlet.ServletException;
     14 import javax.servlet.annotation.WebServlet;
     15 import javax.servlet.http.HttpServlet;
     16 import javax.servlet.http.HttpServletRequest;
     17 import javax.servlet.http.HttpServletResponse;
     18 import javax.servlet.http.HttpSession;
     19 
     20 import sun.java2d.loops.DrawLine;
     21 
     22 /**
     23  * 输出随机的验证码
     24  */
     25 @WebServlet({ "/CheckCode", "/checkCode.jpg" })
     26 public class CheckCodeServlet extends HttpServlet {
     27     private static final long serialVersionUID = 1L;
     28 
     29     /* 宽度 */
     30     private final int WIDTH = 100;
     31     /* 高度 */
     32     private final int HEIGHT = 20;
     33     /* 生成验证码的个数 */
     34     private final int COUNT = 4;
     35     /* 干扰线条数 */
     36     private final int LINE_ROW = 6;
     37 
     38     /* 输出的基本码表,如果使用中文,则使用utf-8的码表,类似 ue234 ,而且应该使用常用字,避免出现偏僻字 */
     39     private final char[] BASECODE = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h',
     40             'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u',
     41             'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H',
     42             'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U',
     43             'V', 'W', 'X', 'Y', 'Z', '0', '1', '2', '3', '4', '5', '6', '7',
     44             '8', '9' };
     45 
     46     // 写出数据
     47     private void write(HttpServletRequest request, HttpServletResponse response)
     48             throws IOException {
     49         HttpSession session = request.getSession();
     50 
     51         BufferedImage image = new BufferedImage(WIDTH, HEIGHT,
     52                 BufferedImage.TYPE_INT_RGB);
     53         Graphics graphics = image.getGraphics();
     54 
     55         setBackground(graphics);
     56         drawBorder(graphics);
     57         drawDot(graphics);
     58         drawLine(graphics);
     59         drawString(graphics, session);
     60 
     61         // 写出数据流
     62         ImageIO.write(image, "jpg", response.getOutputStream());
     63 
     64     }
     65 
     66     // 写字
     67     private void drawString(Graphics graphics, HttpSession session) {
     68         StringBuffer sb = new StringBuffer();
     69         Random random = new Random();
     70 
     71         graphics.setFont(new Font("宋体", Font.BOLD, 18));
     72         graphics.setColor(Color.BLACK);
     73 
     74         for (int i = 0; i < COUNT; i++) {
     75             String ch = String
     76                     .valueOf(BASECODE[random.nextInt(BASECODE.length)]);
     77             sb.append(ch);
     78 
     79             // 设置位置
     80             int x = i * 20 + random.nextInt(12) + 10;
     81             int y = random.nextInt(HEIGHT / 3) + 12;
     82 
     83             // 旋转字体
     84             double theta = Math.PI / 180 * random.nextInt(20);
     85             // rotate(graphics, theta);
     86 
     87             graphics.drawString(ch, x, y);
     88 
     89             // 恢复。。
     90             // rotate(graphics, -theta);
     91         }
     92         session.setAttribute("checkCode", sb.toString());
     93 
     94         System.out.println("session:" + session + "   验证码:" + sb.toString());
     95     }
     96 
     97     // 旋转
     98     private void rotate(Graphics graphics, double theta) {
     99         ((Graphics2D) graphics).rotate(theta);
    100     }
    101 
    102     // 画随机线条
    103     private void drawLine(Graphics graphics) {
    104         Random random = new Random();
    105         for (int i = 0; i < LINE_ROW; i++) {
    106             int x1 = random.nextInt(WIDTH);
    107             int y1 = random.nextInt(HEIGHT);
    108             int x2 = random.nextInt(WIDTH);
    109             int y2 = random.nextInt(HEIGHT);
    110             setRandomColor(graphics);
    111             graphics.drawLine(x1, y1, x2, y2);
    112         }
    113     }
    114 
    115     // 画斑点
    116     private void drawDot(Graphics graphics) {
    117         Random random = new Random();
    118         graphics.setColor(Color.red);
    119         for (int i = 0; i < WIDTH; i++) {
    120             int x = i;
    121             int y = random.nextInt(HEIGHT);
    122             int r = random.nextInt(2);
    123             // graphics.fillOval(x, y, r, r);
    124             graphics.drawOval(x, y, r, r);
    125         }
    126     }
    127 
    128     // 画边框
    129     private void drawBorder(Graphics graphics) {
    130         graphics.setColor(Color.BLACK);
    131         graphics.drawRect(1, 1, WIDTH - 2, HEIGHT - 2);
    132     }
    133 
    134     // 设置背景
    135     private void setBackground(Graphics graphics) {
    136         graphics.setColor(Color.WHITE);
    137         graphics.fillRect(0, 0, WIDTH, HEIGHT);// 填充背景色
    138     }
    139 
    140     // 设置随机的画笔颜色
    141     private void setRandomColor(Graphics g) {
    142         Random random = new Random();
    143         g.setColor(new Color(random.nextInt(255), random.nextInt(255), random
    144                 .nextInt(255)));
    145     }
    146 
    147     protected void doGet(HttpServletRequest request,
    148             HttpServletResponse response) throws ServletException, IOException {
    149 
    150         // 输出图片流的头信息
    151         response.setContentType("image/jpeg");
    152         response.setHeader("Expires", "-1");
    153         response.setHeader("Cache-Control", "no-cache");
    154 
    155         // 写出数据
    156         write(request, response);
    157 
    158     }
    159 
    160     protected void doPost(HttpServletRequest request,
    161             HttpServletResponse response) throws ServletException, IOException {
    162         // TODO Auto-generated method stub
    163     }
    164 
    165 }

    注册提交

    package com.act262.demo;
    
    import java.io.IOException;
    
    import javax.servlet.RequestDispatcher;
    import javax.servlet.ServletException;
    import javax.servlet.annotation.WebServlet;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import javax.servlet.http.HttpSession;
    
    /**
     * Servlet implementation class RegisterServlet
     */
    @WebServlet({ "/RegisterServlet", "/register.do" })
    public class RegisterServlet extends HttpServlet {
        private static final long serialVersionUID = 1L;
    
        protected void doGet(HttpServletRequest request,
                HttpServletResponse response) throws ServletException, IOException {
            HttpSession session = request.getSession();
            String checkCode = (String) session.getAttribute("checkCode");
    
            String paramsName = request.getParameter("name");
            String paramsPwd = request.getParameter("pwd");
            String paramsCheckCode = request.getParameter("checkCode").trim();
    
            // do somethines
    
            //
            response.setContentType("text/html;charset=utf-8");
            if (checkCode.equalsIgnoreCase(paramsCheckCode)) {
                System.out.println("check code ok");
                response.getWriter().write(
                        "尊敬的 <bold>" + paramsName + "</bold>用户,您的密码为:" + paramsPwd
                                + "<br/>验证码 OK");
            } else {
                response.getWriter().write(
                        "尊敬的 <bold>" + paramsName + "</bold>用户,您的密码为:" + paramsPwd
                                + "<br/>验证码 not OK");
            }
        }
    
        protected void doPost(HttpServletRequest request,
                HttpServletResponse response) throws ServletException, IOException {
            doGet(request, response);
        }
    
    }

    jsp简单的界面

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
    <title>Insert title here</title>
    
    
    </head>
    <body>
    
        <div>
    
            <form action="register.do">
                <label>用户名:</label><input type="text" name="name" /></br> <label>密
                    码:</label><input type="password" name="pwd" /></br> <label>验证码:</label><input
                    type="text" name="checkCode" /><img src="checkCode.jpg"
                    onclick=flushCheckCode(this) alt="点击刷新验证码" style="cursor: hand" /></br>
                <input type="submit" value="注册" />
            </form>
    
        </div>
    
        <script type="text/javascript">
            function flushCheckCode(obj) {
                obj.src = (obj.src + '?' + new Date())
            }
        </script>
    </body>
    </html>

    這样子还是可以得,哈哈。。

  • 相关阅读:
    html语法规范
    html页面基本结构
    HTML头部结构详解
    文件路径中 / 和 ./ 和 ../的区别
    实体符号
    利用JS修改style属性和添加元素类名(important)
    Less
    Eureka自我保护计算
    Eureka元数据
    EurekaServer源码分析
  • 原文地址:https://www.cnblogs.com/act262/p/4104387.html
Copyright © 2011-2022 走看看