首先,写一个验证码生成帮助类,用来绘制随机字母:

<span style="font-size:14px;">import java.awt.Color;  
import java.awt.Font;  
import java.awt.Graphics;  
import java.awt.image.BufferedImage;  
import java.io.IOException;  
import java.io.OutputStream;  
import java.util.Random;  
  
import javax.imageio.ImageIO;  
  
public final class GraphicHelper {  
  
    /** 
     * 以字符串形式返回生成的验证码,同时输出一个图片 
     *  
     * @param width 
     *            图片的宽度 
     * @param height 
     *            图片的高度 
     * @param imgType 
     *            图片的类型 
     * @param output 
     *            图片的输出流(图片将输出到这个流中) 
     * @return 返回所生成的验证码(字符串) 
     */  
    public static String create(final int width, final int height, final String imgType, OutputStream output) {  
    StringBuffer sb = new StringBuffer();  
    Random random = new Random();  
  
    BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);  
    Graphics graphic = image.getGraphics();  
  
    graphic.setColor(Color.getColor("F8F8F8"));  
    graphic.fillRect(0, 0, width, height);  
  
    Color[] colors = new Color[] { Color.BLUE, Color.GRAY, Color.GREEN, Color.RED, Color.BLACK, Color.ORANGE,  
        Color.CYAN };  
    // 在 "画板"上生成干扰线条 ( 50 是线条个数)  
    for (int i = 0; i < 50; i++) {  
        graphic.setColor(colors[random.nextInt(colors.length)]);  
        final int x = random.nextInt(width);  
        final int y = random.nextInt(height);  
        final int w = random.nextInt(20);  
        final int h = random.nextInt(20);  
        final int signA = random.nextBoolean() ? 1 : -1;  
        final int signB = random.nextBoolean() ? 1 : -1;  
        graphic.drawLine(x, y, x + w * signA, y + h * signB);  
    }  
  
    // 在 "画板"上绘制字母  
    graphic.setFont(new Font("Comic Sans MS", Font.BOLD, 30));  
    for (int i = 0; i < 6; i++) {  
        final int temp = random.nextInt(26) + 97;  
        String s = String.valueOf((char) temp);  
        sb.append(s);  
        graphic.setColor(colors[random.nextInt(colors.length)]);  
        graphic.drawString(s, i * (width / 6), height - (height / 3));  
    }  
    graphic.dispose();  
    try {  
        ImageIO.write(image, imgType, output);  
    } catch (IOException e) {  
        e.printStackTrace();  
    }  
    return sb.toString();  
    }  
  
}  
</span>  
接着,创建一个servlet,用来固定图片大小,以及处理验证码的使用场景,以及捕获页面生成的验证码(捕获到的二维码与用户输入的验证码一致才能通过)。
[java] view plain copy  
<span style="font-size:14px;">import java.io.IOException;  
import java.io.OutputStream;  
  
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;  
  
@WebServlet(urlPatterns = "/verify/regist.do" )  
public class VerifyCodeServlet extends HttpServlet {  
  
    private static final long serialVersionUID = 3398560501558431737L;  
  
    @Override  
    protected void service(HttpServletRequest request, HttpServletResponse response)  
        throws ServletException, IOException {  
  
    // 获得 当前请求 对应的 会话对象  
    HttpSession session = request.getSession();  
  
    // 从请求中获得 URI ( 统一资源标识符 )  
    String uri = request.getRequestURI();  
    System.out.println("hello : " + uri);  
  
    final int width = 180; // 图片宽度  
    final int height = 40; // 图片高度  
    final String imgType = "jpeg"; // 指定图片格式 (不是指MIME类型)  
    final OutputStream output = response.getOutputStream(); // 获得可以向客户端返回图片的输出流  
                                // (字节流)  
    // 创建验证码图片并返回图片上的字符串  
    String code = GraphicHelper.create(width, height, imgType, output);  
    System.out.println("验证码内容: " + code);  
  
    // 建立 uri 和 相应的 验证码 的关联 ( 存储到当前会话对象的属性中 )  
    session.setAttribute(uri, code);  
  
    System.out.println(session.getAttribute(uri));  
  
    }  
  
}  
</span>  
接着写一个HTML注册页面用来检验一下:
[html] view plain copy  
<span style="font-size:14px;"><!DOCTYPE html>  
<html>  
<head>  
    <meta charset="UTF-8">  
    <title>注册</title>  
    <link rel="stylesheet" href="styles/general.css">  
    <link rel="stylesheet" href="styles/cell.css">  
    <link rel="stylesheet" href="styles/form.css">  
    <script type="text/javascript" src="js/ref.js"></script>  
    <style type="text/css" >  
  
        .logo-container {  
            margin-top: 50px ;  
        }  
        .logo-container img {  
            width: 100px ;  
        }  
  
        .message-container {  
            height: 80px ;  
        }  
  
        .link-container {  
            height: 40px ;  
            line-height: 40px ;  
        }  
  
        .link-container a {  
            text-decoration: none ;  
        }  
  
    </style>  
  
</head>  
<body>  
<div class="container form-container">  
    <form action="/wendao/regist.do" method="post">  
        <div class="form"> <!-- 注册表单开始 -->  
  
            <div class="form-row">  
                        <span class="cell-1">  
                            <i class="fa fa-user"></i>  
                        </span>  
                        <span class="cell-11" style="text-align: left;">  
                            <input type="text" name="username" placeholder="请输入用户名">  
                        </span>  
            </div>  
  
            <div class="form-row">  
                        <span class="cell-1">  
                            <i class="fa fa-key"></i>  
                        </span>  
                        <span class="cell-11" style="text-align: left;">  
                            <input type="password" name="password" placeholder="请输入密码">  
                        </span>  
            </div>  
  
            <div class="form-row">  
                        <span class="cell-1">  
                            <i class="fa fa-keyboard-o"></i>  
                        </span>  
                        <span class="cell-11" style="text-align: left;">  
                            <input type="password" name="confirm" placeholder="请确认密码">  
                        </span>  
            </div>  
  
            <div class="form-row">  
                        <span class="cell-7">  
                            <input type="text" name="verifyCode" placeholder="请输入验证码">  
                        </span>  
                        <span class="cell-5" style="text-align: center;">  
                            <img src="/demo/verify/regist.do" onclick="myRefersh(this)">  
                        </span>  
            </div>  
  
            <div class="form-row" style="border: none;">  
                        <span class="cell-6" style="text-align: left">  
                           <input type="reset" value="重置">  
                        </span>  
                        <span class="cell-6"  style="text-align:right;">  
                            <input type="submit" value="注册">  
                        </span>  
            </div>  
  
        </div> <!-- 注册表单结束 -->  
    </form>  
</div>  
  
</body>  
</html></span>  
效果如下图:

在控制台接收到的图片中验证码的变化如下:

当点击刷新页面的时候,验证码也会随着变化,但我们看不清验证码时,只要点击验证码就会刷新,这样局部的刷新可以用JavaScript来实现。
在<img src="/demo/verify/regist.do">中,添加一个问号和一串后缀数字,当刷新时让后缀数字不断改变,那么形成的验证码也会不断变化,我们可以采用的一种办法是后缀数字用date代替,date获取本机时间,时间是随时变的,这样就保证了刷新验证码可以随时变化。

代码如下:

[javascript] view plain copy  
<span style="font-size:14px;">function myRefersh( e ) {  
      
    const source = e.src ; // 获得原来的 src 中的内容  
    //console.log( "source : " + source  ) ;  
      
    var index = source.indexOf( "?" ) ;  // 从 source 中寻找 ? 第一次出现的位置 (如果不存在则返回 -1 )  
    //console.log( "index : " + index  ) ;  
      
    if( index > -1 ) { // 如果找到了 ?  就进入内部  
        var s = source.substring( 0 , index ) ; // 从 source 中截取 index 之前的内容 ( index 以及 index 之后的内容都被舍弃 )  
        //console.log( "s : " + s  ) ;  
          
        var date = new Date(); // 创建一个 Date 对象的 一个 实例  
        var time = date.getTime() ; // 从 新创建的 Date 对象的实例中获得该时间对应毫秒值  
        e.src = s + "?time=" + time ; // 将 加了 尾巴 的 地址 重新放入到 src 上  
          
        //console.log( e.src ) ;  
    } else {  
        var date = new Date();  
        e.src = source + "?time=" + date.getTime();  
    }  
      
}</span>

---------------------
作者:天堂地址不详Y
来源:CSDN
原文:https://blog.csdn.net/tiantangdizhibuxiang/article/details/77443125

Web后端 JAVA实现验证码生成与验证功能的更多相关文章

  1. EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充

    EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充 (版权声明,本人原创或者翻译的文章如需转载,如转载用于个人学习,请注明出处:否则请与本人联系,违者必究) EO理论上 ...

  2. php 图片验证码生成 前后台验证

    自己从前一段时间做了个php小项目,关于生成图片验证码生成和后台的验证,把自己用到的东西总结一下,希望大家在用到相关问题的时候可以有一定的参考性. 首先,php验证码生成. 代码如下: 1.生成图像代 ...

  3. [转]php 图片验证码生成 前后台验证

    本文转自:https://www.cnblogs.com/xiaoyezi/p/3541195.html 自己从前一段时间做了个php小项目,关于生成图片验证码生成和后台的验证,把自己用到的东西总结一 ...

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

    最近做验证码,参考网上案例,发现有不少问题,特意进行了修改和完善. 验证码生成器: import javax.imageio.ImageIO; import java.awt.*; import ja ...

  5. struts---JSP界面验证码生成与验证

    之前想做一个随机验证码的功能,自己也搜索了一下别人写的代码,然后自己重新用struts2实现了一下,现在将我自己实现代码贴出来!大家有什么意见都可以指出来! 首先是生成随机验证码图片的action: ...

  6. java模拟验证码生成

    设计思想 第一步:随机生成字符串 第二步:用户输入字符串 第三步:将两个字符串转化为同一类型 第四步:比较是否相同 第五步:输出结果 程序流程图 程序源代码 /*2017/10/7 王翌淞 验证码模拟 ...

  7. Java图片验证码生成工具

    直接把以下代码拷贝使用: import javax.imageio.ImageIO;import java.awt.*;import java.awt.image.BufferedImage;impo ...

  8. PHP 用session与gd库实现简单验证码生成与验证的类

    验证码是为了防止机器灌水给网站带来污染以及增加服务器负担而出现的.目前大大小小的网站都有验证码.今天自己实现了一个简单的验证码类.说简单是因为没有加一些干扰的弧线等等,只是将文字旋转了一下.当然,因为 ...

  9. 一篇Java图片验证码生成的代码

    package projectUtil; /** * @author tian * @date 2019/4/1015:58 */ import javax.imageio.ImageIO; impo ...

随机推荐

  1. 【STM32H7教程】第9章 STM32H7重要知识点数据类型,变量和堆栈

    完整教程下载地址:http://forum.armfly.com/forum.php?mod=viewthread&tid=86980 第9章   STM32H7重要知识点数据类型,变量和堆栈 ...

  2. CTF丨2019互联网安全城市巡回赛·西安站,我们来了!

    万物互联时代,网信事业发展突飞猛进,互联网悄然渗透到国民生活的每一个角落,伴随而来的网络安全威胁和风险也日渐突出.网络诈骗.钓鱼软件.勒索病毒等安全问题层出不穷,信息泄露等网络安全事件也频繁上演,给用 ...

  3. Linux上删除大量文件几种方式对比

    目录 Linux上删除大量文件几种方式对比 1. rm删除:因为文件数量太多,rm无法删除(报错) 2. find查找删除:-exec 3. find查找删除:xargs 4. find调用-dele ...

  4. redis测试实践

    最近测试服务端的时候,接触到了redis,之前也看过,但不系统,借着这次实践,记录一下. 一.写在前面 Redis是一个开源的使用ANSI C语言编写.遵守BSD协议.支持网络.可基于内存亦可持久化的 ...

  5. 带着新人看java虚拟机05(多线程篇)

    上一篇我们主要是把一些基本概念给说了一下以及怎么简单的使用线程池,我们这一节就来看看线程池的实现: 1.线程池基本参数 以Executors.newFixedThreadPool()这种创建方式为例: ...

  6. [译]PEP 380--子生成器的语法

    导语: PEP(Python增强提案)几乎是 Python 社区中最重要的文档,它们提供了公告信息.指导流程.新功能的设计及使用说明等内容.对于学习者来说,PEP 是非常值得一读的第一手材料,学习中遇 ...

  7. C#设计模式整理

    我居然连一月一随笔都没有,啊啊啊,忙死个人 这个随笔主要是记录基于自己学习[美]James W.Cooper著的<C# Design Patterns : A Tutorial>一书中常用 ...

  8. UiPath针对SAP的输入技巧

    我观察到在SAP中不论是SimulateType,还是SendWindowMessages,Type Into的输入速度都很慢(是逐个字符输入的).如果只是一次两次的输入倒也没什么,但如果是需要批量多 ...

  9. springmvc重定向

    String success="07大吉大利25"; @RequestMapping("str") public String test1(){ return ...

  10. 关于ORACLE的SQL语句拼接、替换、截取、排序,联表等...~持续汇总~

     先看一下所有的数据.这里全部为VARCHAR2(255). 字段拼接 在所有的性别后面加% 字段替换,把性别TPF_SEX去除百分号% 字段截取 字段截取+拼接 字段替换,这里把百分号%替换为空,也 ...