文档章节

JFinal CaptchaRender图型验证码的又一个实现。

玛雅牛
 玛雅牛
发布于 2013/07/07 00:59
字数 1222
阅读 4156
收藏 40

基于jfinal源代码中的com.jfinal.ext.render.CaptchaRender改写而来。同时参考了@27号 代码

做了以下改变:
1)去掉高度和宽度设定,改为高度固定,宽度依据字符个数自动计算。
2)默认对大小写不敏感。
3)验证码及其md5散列由渲染时声称变为初始化时生成。
4)添加获取md5散列的方法。

基本思路:
CaptchaRender 初始化时同时生成验证码以及md5散列后的验证码,CaptchaRender添加一个方法可以i获取到md5散列后的验证码,这个md5散列后的验证码的由调用方法保存,想存哪里就存哪里,session,cooki,分布式cache中都行,这个md5散列后的验证码也可以发给客户端,通过MD5.js 来进行浏览器端的验证码校验,控制权交给调用方。

上代码:

 

package com.jfinal.ext.render;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.security.MessageDigest;
import java.util.Random;

import javax.imageio.ImageIO;
import javax.servlet.ServletOutputStream;

import com.jfinal.kit.StringKit;
import com.jfinal.render.Render;

/**
 * 验证码Render,这个验证码Render在构造函数里就已经创建好了随机码以及md5散列后的随机码。
 * 调用方式如下:
 * CaptchaRender captchaRender = new CaptchaRender();
 * String md5RandonCode = captchaRender.getMd5RandonCode();
 * 保存md5RandonCode到session、cookie或者其他地方
 * render(captchaRender);
 * 基于JFinal的版本修改。
 *
 */
public class CaptchaRender extends Render
{
	private static final long serialVersionUID = -7599510915228560611L;

	/**
	 * 随机码生成字典
	 */
	private static final String[] strArr = {"3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "J", "K", "M", "N", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y"};


	/**
	 * 默认存储时使用的key,将md5散列后的随机码保存至session,cookie时使用。
	 */
	public static final String DEFAULT_CAPTCHA_MD5_CODE_KEY = "_CAPTCHA_MD5_CODE_";

	/**
	 * 图片宽度
	 */
	private final int imgWidth;

	/**
	 * 图片高度
	 */
	private final int imgHeight;

	/**
	 * 随机生成字符数量
	 */
	private final int imgRandNumber;

	/**
	 * 生成的随机码
	 */
	private final String randonCode;

	/**
	 * md5散列后的随机码
	 */
	private final String md5RandonCode;

	/**
	 * 构造函数,随机生成6个字符。
	 */
	public CaptchaRender() {
		this(6);
	}

	/**
	 * 构造函数
	 * @param imgRandNumber 随机生成多少个字符,最少4个字符。
	 */
	public CaptchaRender(int imgRandNumber) {
		if(imgRandNumber < 4)
		{
			imgRandNumber = 4;
		}
		this.imgWidth = 16*imgRandNumber + 12;
		this.imgHeight = 26;
		this.imgRandNumber = imgRandNumber;
		this.randonCode = generateRandonCode();
		this.md5RandonCode = encrypt(randonCode);
	}

	/**
	 * 获取md5散列后的验证码,调用发需妥善保存此验证码。
	 * @return md5散列后的验证码
	 */
	public String getMd5RandonCode(){
		return this.md5RandonCode;
	}

	/**
	 * 依据字典生成随即码
	 * @return 随机码
	 */
	private String generateRandonCode(){
		// 生成随机类
		Random random = new Random();
		String sRand = "";
		for (int i = 0; i < imgRandNumber; i++) {
			String rand = String.valueOf(strArr[random.nextInt(strArr.length)]);
			sRand += rand;
		}
		return sRand;
	}

	/**
	 * 渲染图片
	 */
	public void render() {
		BufferedImage image = new BufferedImage(imgWidth, imgHeight, BufferedImage.TYPE_INT_RGB);
		drawGraphic(image);
		response.setHeader("Pragma","no-cache");
        response.setHeader("Cache-Control","no-cache");
        response.setDateHeader("Expires", 0);
        response.setContentType("image/jpeg");

        ServletOutputStream sos = null;
        try {
			sos = response.getOutputStream();
			ImageIO.write(image, "jpeg",sos);
		} catch (Exception e) {
			throw new RuntimeException(e);
		}
		finally {
			if (sos != null)
				try {sos.close();} catch (IOException e) {e.printStackTrace();}
		}
	}

	/**
	 * 绘制验证码
	 * @param image BufferedImage对象
	 */
	private void drawGraphic(BufferedImage image){
		// 获取图形上下文
		Graphics g = image.createGraphics();
		// 生成随机类
		Random random = new Random();
		// 设定背景色
		g.setColor(getRandColor(200, 250));
		g.fillRect(0, 0, imgWidth, imgHeight);
		// 设定字体
		g.setFont(new Font("Times New Roman", Font.PLAIN, 18));

		// 随机产生155条干扰线,使图象中的认证码不易被其它程序探测到
		g.setColor(getRandColor(160, 200));
		for (int i = 0; i < 155; i++) {
			int x = random.nextInt(imgWidth);
			int y = random.nextInt(imgHeight);
			int xl = random.nextInt(12);
			int yl = random.nextInt(12);
			g.drawLine(x, y, x + xl, y + yl);
		}

		g.setFont(new Font("TimesRoman", Font.PLAIN, 20));
		//取随机产生的认证码(img_randNumber位数字)
		for (int i = 0; i < imgRandNumber; i++) {
			String rand = String.valueOf(this.randonCode.charAt(i));
			// 将认证码显示到图象中
			g.setColor(new Color(20 + random.nextInt(110), 20 + random.nextInt(110), 20 + random.nextInt(110)));
			// 调用函数出来的颜色相同,可能是因为种子太接近,所以只能直接生成
			g.drawString(rand, 16 * i + 6, 21);
		}
		// 图象生效
		g.dispose();
	}

	/**
	 * 生成随机颜色
	 * @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);
	}

	/**
	 * 使用md5散列字符串
	 * @param srcStr 输入的字符串
	 * @return 加密后的字符串
	 */
	private static final String encrypt(String srcStr) {
		try {
			String result = "";
			MessageDigest md = MessageDigest.getInstance("MD5");
			byte[] bytes = md.digest(srcStr.getBytes("utf-8"));
			for(byte b:bytes){
				String hex = Integer.toHexString(b&0xFF).toUpperCase();
				result += ((hex.length() ==1 ) ? "0" : "") + hex;
			}
			return result;
		} catch (Exception e) {
			throw new RuntimeException(e);
		}
	}

	/**
	 * 验证码检查
	 * @param md5RandomCode  md5散列后的验证码
	 * @param inputRandomCode 用户输入的验证码
	 * @return 若二者一致,返回true,否则返回false
	 */
	public static boolean validate(String md5RandomCode, String inputRandomCode) {
		if (StringKit.isBlank(md5RandomCode) || StringKit.isBlank(inputRandomCode))
			return false;
		try {
			inputRandomCode = inputRandomCode.toUpperCase();
			inputRandomCode = encrypt(inputRandomCode);
			return inputRandomCode.equals(md5RandomCode);
		} catch (Exception e) {
			e.printStackTrace();
			return false;
		}
	}

}

 

action代码如下:

CaptchaRender img = new CaptchaRender(4); this.setSessionAttr(CaptchaRender.DEFAULT_CAPTCHA_MD5_CODE_KEY, img.getMd5RandonCode());
render(img);

页面代码,点击自动刷新:

 

<script type="text/javascript">
	function refushcode(){
		var d = new Date();
		//为了避免服务器或者浏览器缓存,添加了一个额外的参数
		document.getElementById("safecode").src="${BASE_PATH}/au/img?t="+d.toString(40);
	}
   </script>

<img src="${BASE_PATH}/au/img" alt="点击刷新" style="padding-top: 7px;" id="safecode" onclick="refushcode();"/>

验证action代码:

 

String captchaCode = getPara("captchaCode");
Object objMd5RandomCode = this.getSessionAttr(CaptchaRender.DEFAULT_CAPTCHA_MD5_CODE_KEY);
			String md5RandomCode = null;
			if(objMd5RandomCode != null){
				md5RandomCode = objMd5RandomCode.toString();
				this.removeSessionAttr(CaptchaRender.DEFAULT_CAPTCHA_MD5_CODE_KEY);
			}
			if(!CaptchaRender.validate(md5RandomCode, captchaCode)){
				this.setFlash("username", username);
				this.setFlash("password", password);
				this.setFlash("msg", "验证码错误,请输入正确的验证码");
				redirect(ShiroKit.getLoginUrl());
				return;
			}

打完收工。

© 著作权归作者所有

共有 人打赏支持
玛雅牛

玛雅牛

粉丝 483
博文 113
码字总数 27287
作品 4
高级程序员
私信 提问
加载中

评论(7)

玛雅牛
玛雅牛

引用来自“i-bdyr”的评论

CaptchaRender img = new CaptchaRender(4);
String cookieName =CaptchaRender.DEFAULT_CAPTCHA_MD5_CODE_KEY;
String cookieVal = img.getMd5RandonCode();

Cookie cookie = new Cookie(cookieName,cookieVal);
this.setCookie(cookie);

render(img);

验证的时候从cookie 中获取不到啊
我一般是存在session中。没有给cookie放过。你可尝试放到session中。
badouyuren
badouyuren
CaptchaRender img = new CaptchaRender(4);
String cookieName =CaptchaRender.DEFAULT_CAPTCHA_MD5_CODE_KEY;
String cookieVal = img.getMd5RandonCode();

Cookie cookie = new Cookie(cookieName,cookieVal);
this.setCookie(cookie);

render(img);

验证的时候从cookie 中获取不到啊
玛雅牛
玛雅牛

引用来自“简单代码”的评论

原来大牛也是陕西的呀。。。幸会幸会~~83
嗯。西安人。幸会。
简单代码
简单代码
原来大牛也是陕西的呀。。。幸会幸会~~83
小强哥unas
小强哥unas
非常和我的心意
骑着瓜牛看河蟹
骑着瓜牛看河蟹
fdsfdsfds
okk
okk
zan,bu cuo !
JFinal中使用CaptchaRender来实现图型验证码

本人文字功底有限,直接上代码,此代码是参考jfinal源代码中的com.jfinal.ext.render.CaptchaRender,改写而来,为什么要改写?因为这个类,作者目前的实现还不满足要求,所以,自己就改了下...

27号
2012/12/21
0
4
JFinal CaptchaRender的简单使用

稍微修改了下jfinal官方demo的代码,用了下验证码 效果图: CommonController.java package com.demo.common; import com.jfinal.core.Controller;import com.jfinal.lib.captcha.CaptchaRe......

绝望的八皮
2012/07/06
0
13
jfinal+shiro基础上添加jcaptcha验证码

http://www.oschina.net/code/snippet13398932527 上门链接是上次分享了在jfinal中添加jcaptcha验证码 后看到@玛雅牛的jfinal-shiro插件,顺便将验证码加入 MyShiroRealm.java /** * 认证回调...

edeis2011
2014/01/10
0
4
JFinal极速开发实战教程新鲜出炉~

JFinal极速开发实战教程新鲜出炉~ [持续更新中...] JFinal框架让 IT 工程师发挥更大的潜能,实现更大价值,同时也希望 JFinal 工具力量能帮助正在创业的朋友们提升效率、降低成本从而实现创业...

山东-小木
2015/01/07
0
13
JFinal的Model类的二次继承方法

我想要扩展以下JFinal的Model的功能,先继承这个基类为BaseModel,然后所有的POJO都继承这个BaseModel,这样就很方便扩展在Model中的功能,又不改变Model的代码,方便扩展自己的功能。 查看了...

Jetmark
2013/05/12
0
3

没有更多内容

加载失败,请刷新页面

加载更多

Redis异构集群之间数据迁移方案

一、Redis集群迁移工具 最近在做Redis数据迁移,网上找了两款开源的Redis迁移工具。 第一种:redis-port Codis官方提供的一个工具,redis-port是一个Redis工具,通过解析rdb文件,实现Redis...

IT--小哥
21分钟前
1
0
MySQL5.7源码安装

5、MySQL源码安装(centos7.5) 5.1 下载软件 官方自带(boost版本mysql) wget https://dev.mysql.com/get/Downloads/MySQL-5.7/mysql-boost-5.7.20.tar.gz tar xf mysql-boost-5.7.20.tar.g......

hnairdb
25分钟前
1
0
解压zip包(zip4j)

1:引入zip4j_1.3.2.jar 2:源码如下: package test;import java.io.File;import java.io.IOException;import java.util.ArrayList;import java.util.List;import net.lingal......

uug
28分钟前
1
0
MySQL 8在CentOS 7用rpm安装

1.配置MySQL yum安装源和安装: #配置MySQL5.7的安装源 #https://repo.mysql.com/mysql57-community-release-el7.rpm #https://repo.mysql.com/mysql57-community-release-el7-11.noarch.rp......

MichaelShu
35分钟前
1
0
docker 安装zookeeper

1.下拉镜像: docker pull zookeeper 2.运行镜像: docker run --name myZookeeper --restart always -d zookeeper 此时,2181 2888 3888分别是zookeeper的(客户端端口,跟随端口,选择端口)...

狼王黄师傅
36分钟前
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部