基于Servlet的验证码实现

话不多说先上代码

public class VerCodeServlet extends HttpServlet {

    private static final long serialVersionUID = 5126616339795936447L;

    private ConfigurableCaptchaService configurableCaptchaService = null;
    private RandomFontFactory fontFactory = null;
    private RandomWordFactory wordFactory = null;
    private TextRenderer textRenderer = null;

    public VerCodeServlet() {
        super();
    }

    /**
     * doGet
     */
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }

    /**
     * doPost 输出图片流
     */
    public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        Dto inDto =Dtos.newInDto(request); //WebCxt.getParamAsDto(request);
        doConfig(inDto); // 需要动态配置的参数

        response.setContentType("image/png");
        response.setHeader("Cache-Control", "private, no-store, no-cache, must-revalidate"); // HTTP
        response.setHeader("Pragma", "no-cache"); // HTTP 1.0

        OutputStream outputStream = response.getOutputStream();

        // 得到验证码对象,有验证码图片和验证码字符串
        Captcha captcha = configurableCaptchaService.getCaptcha();
        // 取得验证码字符串放入Session
        String verificationCode = captcha.getChallenge();
        request.getSession().setAttribute(Constant.VERCODE, verificationCode);
        //JedisUtil.setString(inDto.getString("uuid"), verificationCode, 5 * 60);
        // 取得验证码图片并输出
        BufferedImage bufferedImage = captcha.getImage();
        ImageIO.write(bufferedImage, "png", outputStream);

        outputStream.flush();
        outputStream.close();
    }

    /**
     * 需要动态配置的参数
     */
    private void doConfig(Dto inDto) {
        // 随机字体生成器
        fontFactory = new RandomFontFactory();
        int fontSize = 25;
        if (AOSUtils.isNotEmpty(inDto.getString("fontSize"))) {
            fontSize = new Integer(inDto.getString("fontSize")).intValue();
        }
        fontFactory.setMaxSize(fontSize);
        fontFactory.setMinSize(fontSize);
        configurableCaptchaService.setFontFactory(fontFactory);

        // 随机字符生成器,去除掉容易混淆的字母和数字,如o和0等
        wordFactory = new RandomWordFactory();
        String characters = "abcdefhkty2345678";
        if (AOSUtils.isNotEmpty(inDto.getString("characters"))) {
            characters = inDto.getString("characters");
        }
        wordFactory.setCharacters(characters);
        Integer length = Integer.valueOf(4);
        if (AOSUtils.isNotEmpty(inDto.getString("length"))) {
            length = Integer.valueOf((inDto.getString("length")));
        }
        wordFactory.setMaxLength(length);
        wordFactory.setMinLength(length);
        configurableCaptchaService.setWordFactory(wordFactory);

        // 验证码图片的大小
        Integer width = Integer.valueOf(100);
        if (AOSUtils.isNotEmpty(inDto.getString("width"))) {
            width = Integer.valueOf((inDto.getString("width")));
        }
        Integer height = Integer.valueOf(40);
        if (AOSUtils.isNotEmpty(inDto.getString("height"))) {
            height = Integer.valueOf((inDto.getString("height")));
        }
        configurableCaptchaService.setWidth(width);
        configurableCaptchaService.setHeight(height);
    }

    /**
     * Servlet初始化方法:只会初始化一次。如果要做相关动态处理,请将代码移到doPost。
     */
    public void init() throws ServletException {
        configurableCaptchaService = new ConfigurableCaptchaService();

        // 颜色创建工厂
        // 单色的
        // configurableCaptchaService.setColorFactory(new
        // SingleColorFactory(newColor(25, 60, 170)));
        // 彩色的
        configurableCaptchaService.setColorFactory(new ColorFactory() {
            private Random random = new Random();

            @Override
            public Color getColor(int x) {
                int[] c = new int[3];
                int i = random.nextInt(c.length);
                for (int fi = 0; fi < c.length; fi++) {
                    if (fi == i) {
                        c[fi] = random.nextInt(71);
                    } else {
                        c[fi] = random.nextInt(256);
                    }
                }
                return new Color(c[0], c[1], c[2]);
            }
        });

        // 自定义验证码图片背景
        MyCustomBackgroundFactory backgroundFactory = new MyCustomBackgroundFactory();
        configurableCaptchaService.setBackgroundFactory(backgroundFactory);

        // 图片滤镜设置
        ConfigurableFilterFactory filterFactory = new ConfigurableFilterFactory();
        List<BufferedImageOp> filters = new ArrayList<BufferedImageOp>();
        WobbleImageOp wobbleImageOp = new WobbleImageOp();
        wobbleImageOp.setEdgeMode(AbstractImageOp.EDGE_CLAMP);
        wobbleImageOp.setxAmplitude(1.0);
        wobbleImageOp.setyAmplitude(2.0);
        filters.add(wobbleImageOp);
        filterFactory.setFilters(filters);
        configurableCaptchaService.setFilterFactory(filterFactory);

        // 文字渲染器设置
        textRenderer = new BestFitTextRenderer();
        textRenderer.setBottomMargin(1);
        textRenderer.setTopMargin(1);
        configurableCaptchaService.setTextRenderer(textRenderer);
    }

    /**
     * Servlet销毁方法,负责销毁所使用资源.
     */
    public void destroy() {
        wordFactory = null;
        fontFactory = null;
        textRenderer = null;
        configurableCaptchaService = null;
        super.destroy();
    }

    /**
     * 内部类 自定义验证码图片背景,主要画一些噪点和干扰线
     */
    private class MyCustomBackgroundFactory implements BackgroundFactory {
        private Random random = new Random();

        public void fillBackground(BufferedImage image) {
            Graphics graphics = image.getGraphics();

            // 验证码图片的宽高
            int imgWidth = image.getWidth();
            int imgHeight = image.getHeight();

            // 填充为白色背景
            graphics.setColor(Color.white);
            graphics.fillRect(0, 0, imgWidth, imgHeight);

            // 画100个噪点(颜色及位置随机)
            for (int i = 0; i < 25; i++) {
                // 随机颜色
                int rInt = random.nextInt(255);
                int gInt = random.nextInt(255);
                int bInt = random.nextInt(255);

                graphics.setColor(new Color(rInt, gInt, bInt));

                // 随机位置
                int xInt = random.nextInt(imgWidth - 3);
                int yInt = random.nextInt(imgHeight - 2);

                // 随机旋转角度
                int sAngleInt = random.nextInt(360);
                int eAngleInt = random.nextInt(360);

                // 随机大小
                int wInt = random.nextInt(6);
                int hInt = random.nextInt(6);

                graphics.fillArc(xInt, yInt, wInt, hInt, sAngleInt, eAngleInt);

                // 画干扰线
                if (i % 5 == 0) {
                    int xInt2 = random.nextInt(imgWidth);
                    int yInt2 = random.nextInt(imgHeight);
                    graphics.drawLine(xInt, yInt, xInt2, yInt2);
                }
            }
        }
    }
}

生成验证码图片,并保存验证码在session中等待校验(以上代码会依赖AOS框架中的类需引入).

下面是servlet配置

<!-- 验证码Servlet -->
<servlet>
<servlet-name>verCodeServlet</servlet-name>
<servlet-class>com.******.common.VerCodeServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>verCodeServlet</servlet-name>
<url-pattern>/verCodeServlet.png</url-pattern>
</servlet-mapping>

下面是JSP代码

<img id="verCode" src="${pageContext.request.contextPath}/verCodeServlet.png"> <a id="kanbuq" href="javascript:void(0);" onclick="changeVer()">看不清,换一张</a> </div>

function changeVer(){
$("#verCode").attr(‘src‘,‘${pageContext.request.contextPath}/verCodeServlet.png‘);
}

验证代码就不上了。

时间: 2024-08-06 21:09:17

基于Servlet的验证码实现的相关文章

基于Servlet的技术问答网站系统实现

这一篇博客将详细介绍一个基于Servlet的问答网站的实现,有详细的代码. 可能篇幅较长,以代码为主,有兴趣的童鞋看完可以尝试动手搭建一个属于自己的问答社区. 工具:Eclipse,数据库用到了MySQL,这次项目中未使用jsp,全部以Servlet注解的方式连接HTML和Servlet,JDK最好使用1.8,tomcat使用8.0.(注解方式为JDK1.5后的特性,最低要求1.5+,本项目使用JDK1.8). 在这篇博客中可以学习到: 1,Servlet中关于注解的使用,本项目没有使用到传统的

基于servlet实现一个web框架

servlet作为一个web规范.其本身就算做一个web开发框架,可是其web action (响应某个URI的实现)的实现都是基于类的,不是非常方便,而且3.0之前的版本号还必须通过web.xml配置来添加新的action. servlet中有一个filter的功能,能够配置全部URI的功能都经过filter.我们能够基于filter的功能来实现一个简单的web框架.在这个框架中,主要改进URI action的映射,就像play framework中route的配置: GET /hello c

转: 通过Servlet生成验证码图片

孤傲苍狼 只为成功找方法,不为失败找借口! javaweb学习总结(九)—— 通过Servlet生成验证码图片 一.BufferedImage类介绍 生成验证码图片主要用到了一个BufferedImage类,如下: 创建一个DrawImage Servlet,用来生成验证码图片 1 package gacl.response.study; 2 import java.awt.Color; 3 import java.awt.Font; 4 import java.awt.Graphics; 5

基于Servlet、JSP、JDBC、MySQL的登录模块(含过滤器的使用和配置)

接前文的注册模块,本篇是登录模块.主要包括登录主界面,和登录相关编写的LoginAction.LoginDao和LoginService.以及配置的Filter.下面按逻辑顺序记录详细过程和代码: 一.在前文的index目录点击登录按钮后,通过javascript跳转至LoginAction. <script type="text/javascript"> function login(){ var th = document.form1; if(th.username.v

servlet生成验证码验证的实现

servlet中的方法实现代码: import java.awt.Color; import java.awt.Font; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.IOException; import java.io.PrintWriter; import java.util.Random; import javax.servlet.ServletException; imp

对基于Servlet+JSP+JavaBean开发模式的用户登录注册的升级

还记得我前面所写的博文基于Servlet+JSP+JavaBean开发模式的用户登录注册吗?我们以前是创建代表数据库的xml文件来保存用户信息的,现在我们已经学习了数据库相关的知识,所以应把xml换成数据库,升级成数据库应用. 我们在把以前的工程复制并拷贝时,假设以前的工程名是day09_user,现复制一份并拷贝,重新修改工程名为day14_user,此刻将其直接部署在tomcat服务器上,那么day14_user这个JavaWeb应用映射的虚拟目录仍然是"/day09_user",

基于Servlet、JSP、JDBC、MySQL的一个简单的用户注册模块(附完整源码)

最近看老罗视频,做了一个简单的用户注册系统.用户通过网页(JSP)输入用户名.真名和密码,Servlet接收后通过JDBC将信息保存到MySQL中.虽然是个简单的不能再简单的东西,但麻雀虽小,五脏俱全,在此做一归纳和整理.下面先上源码: 一.index.jsp <%@ page language="java" import="java.util.*" pageEncoding="utf-8"%> <% String path =

基于Servlet+smartUpload的文件上传

文件上传在web应用中是非常常见的,现在我就介绍下基于servlet的文件上传,基于Struts2的文件上传可以看: 页面端代码: <%@ page language="java" import="java.util.*" pageEncoding="GBK"%> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <ht

基于Servlet+JDBC+Bootstrap+MySQL+AJAX权限管理系统项目实战教程

项目简介 权限系统一直以来是我们应用系统不可缺少的一个部分,若每个应用系统都重新对系统的权限进行设计,以满足不同系统用户的需求,将会浪费我们不少宝贵时间,所以花时间来设计一个相对通用的权限系统是很有意义的.     本系统的设计目标是对应用系统的所有资源进行权限控制,比如应用系统的功能菜单.各个界面等进行权限的操控.技术介绍 · Servlet3.0 Servlet 3.0 作为JavaEE6 规范体系中一员,随着JavaEE6规范一起发布.该版本在前一版本(Servlet2.5)的基础上提供了