字母数字、字母、汉字验证码 (java)


一、字母数字,字母,汉字验证码的生成代码

1、字母数字验证码:

package com.soufun;
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.util.Random;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
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 ImgVarificationCodeServlet
 */
<span style="color:#FF0000;">// 这里也可以使用web.xml 来配置servlet</span>
@WebServlet("/ImgVarificationCodeServlet")
public class ImgVarificationCodeServlet extends HttpServlet {

	private static final long serialVersionUID = 1L;

   //这里的doGet,doPost都可以不要, 只要有service 就可以
	protected void doPost(HttpServletRequest request,
	HttpServletResponse response) throws ServletException, IOException {
	doGet(request, response);
	}


	protected void doGet(HttpServletRequest request,
	HttpServletResponse response) throws ServletException, IOException {
	}

   /**
    * 定义验证码图片的一些属性
    */
	// 验证码图片的宽度。
	private int width = 60;

	// 验证码左右边距宽度。
	private int padding = 10;

	// 验证码图片的高度。
	private int height = 20;

	// 验证码字符个数
	private int codeCount = 4;
  
	//验证码之间的距离
	private int x = 0;

	// 字体高度
	private int fontHeight;


	private int codeY;


	// 排除容易混淆的字母和数字 I,i,l,L,1,O,o,0-- 验证码显示字符
	char[] codeSequence = { 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'J', 'K',
	'M', 'N', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
	'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'j', 'k', 'm', 'n', 'p',
	'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '2', '3', '4',
	'5', '6', '7', '8', '9' };


	/**
	* 初始化验证图片属性
	*/
	public void init() throws ServletException {
	// 从web.xml中获取初始信息
	// 宽度
	String strWidth = this.getInitParameter("width");
     
	// 高度
	String strHeight = this.getInitParameter("height");
	// 字符个数
	String strCodeCount = this.getInitParameter("codeCount");


	// 将配置的信息转换成数值
	try {
	if (null !=strWidth&& strWidth.length() != 0) {
	width = Integer.parseInt(strWidth);
	if (strHeight != null && strHeight.length() != 0) {
	height = Integer.parseInt(strHeight);
	}
	if (strCodeCount != null && strCodeCount.length() != 0) {
	codeCount = Integer.parseInt(strCodeCount);
	}
	}
	} catch (NumberFormatException e) {

	}


	if (codeCount > 0) {
	x = (width - padding * 2) / (codeCount);
	} else {
	x = 0;
	}
	// 字体高度=图片高度-2
	fontHeight = height - 2;
	// 验证码前在y轴方向上预留的高度
	codeY = height - 4;


	}


	protected void service(HttpServletRequest req, HttpServletResponse resp)
	throws ServletException, java.io.IOException {


	//具有 8 位 RGB 颜色分量的图像 对象
	BufferedImage buffImg = new BufferedImage(width, height,
	BufferedImage.TYPE_INT_RGB);
	// 创建一个 Graphics2D,可以将它绘制到此 BufferedImage 中。
	Graphics2D g = buffImg.createGraphics();

	//创建一个随机数生成器类
	Random random = new Random();

	//调用父类的setColor()方法使此图形上下文的当前颜色设置为指定颜色。使用此图形上下文的所有后续图形操作均使用这个指定的颜色
	g.setColor(Color.WHITE);
	//填充指定的矩形。该矩形左边缘和右边缘分别位于 x 和 x + width - 1。上边缘和下边缘分别位于 y 和 y + height - 1。
	g.fillRect(0, 0, width, height);
	
	//创建字体,字体的大小应该根据图片的高度来定。
	Font font = new Font("System", Font.PLAIN, fontHeight);
	//将此图形上下文的字体设置为指定字体。使用此图形上下文的所有后续文本操作均使用此字体。
	g.setFont(font);


	// 画边框。
	g.setColor(Color.BLACK);
	//绘制指定矩形的边框
	g.drawRect(0, 0, width - 1, height - 1);

	// 随机产生30条干扰线,使图象中的认证码不易被其它程序探测到。
	g.setColor(Color.lightGray);
	for (int i = 0; i < 30; i++) {
	int x = random.nextInt(width);
	int y = random.nextInt(height);
	int xl = random.nextInt(12);
	int yl = random.nextInt(12);
	// 在此图形上下文的坐标系中,使用当前颜色在点 (x1, y1) 和 (x2, y2) 之间画一条线。
	g.drawLine(x, y, x + xl, y + yl);
	}


	// randomCode用于保存随机产生的验证码,以便用户登录后进行验证。
	StringBuffer randomCode = new StringBuffer();
	int red = 0, green = 0, blue = 0;


	// 随机产生codeCount数字的验证码。
	for (int i = 0; i < codeCount; i++) {
	// 得到随机产生的验证码数字。
	String strRand = String.valueOf(codeSequence[random
	.nextInt(codeSequence.length)]);
	// 产生随机的颜色分量来构造颜色值,这样输出的每位数字的颜色值都将不同。
	red = random.nextInt(255);
	green = random.nextInt(255);
	blue = random.nextInt(255);
	// 用随机产生的颜色将验证码绘制到图像中。
	g.setColor(new Color(red, green, blue));
	int codeX = i * x + padding; // 字符前面预留的宽度 (i + 1) * x
	// 到这里就生成了一张有背景色,有边框,有指定个数的验证码被画在了指定的位置并且每个字的颜色不同。
	g.drawString(strRand, codeX, codeY);

	// 将产生的四个随机数组合在一起。
	randomCode.append(strRand);
	}

	// 将四位数字的验证码保存到MemCached中,这样比session好处:1.无域名或者session限制;2.服务器重启不受影响;3.减轻服务器压力
	String sMemValue = "";
    // 验证码
	sMemValue = randomCode.toString();

	//将四位数字的验证码保存到Session中。
   	HttpSession session = req.getSession();
	session.setAttribute("validateCode",sMemValue);
	// 禁止图像缓存。
	resp.setHeader("Pragma", "no-cache");
	resp.setHeader("Cache-Control", "no-cache");
	resp.setDateHeader("Expires", 0);
    // 响应内容类型 image/jpeg
	resp.setContentType("image/jpeg");
	// 将图像输出到Servlet输出流中。
	ServletOutputStream sos = resp.getOutputStream();
	ImageIO.write(buffImg, "jpeg", sos);
	// 释放资源
	sos.close();
	}
}

2、字母验证码 (来自网络):

package com.soufun;

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

import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
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 AuthImageServlet
 */
<span style="color:#FF0000;">// 同上这里可以使用web.xml 配置servlet</span>
@SuppressWarnings("unused")
@WebServlet("/AuthImageServlet")
public class AuthImageServlet extends HttpServlet {
	 
	  
	
	private static final String CONTENT_TYPE = "text/html; charset=gb2312";     
    // 创建字体   
    private Font mFont = new Font("Times New Roman", Font.PLAIN, 17);     
    public void init() throws ServletException     
    {     
        super.init();     
    }     
    // 获取颜色
    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);     
    }     
    
    public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException     
    {     
        response.setHeader("Pragma","No-cache");     
        response.setHeader("Cache-Control","no-cache");     
        response.setDateHeader("Expires", 0);     
        //生成的响应是图片     
        response.setContentType("image/jpeg");     
        // 设置宽度和高度,其实上面的字体的大小和这里的高度有关     
        int width=100, height=18; 
      //具有 8 位 RGB 颜色分量的图像 对象
        BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);     
       //创建一个 Graphics,就是一个画布。之后的操作都在这个画布上进行。
       //可以将它绘制到此 BufferedImage 中。      
        Graphics g = image.getGraphics();     
        Random random = new Random(); 
        // 设置画布的背景颜色
        g.setColor(getRandColor(200,250));   
        //填充指定的矩形画布。该矩形左边缘和右边缘分别位于 x 和 x + width - 1。上边缘和下边缘分别位于 y 和 y + height - 1。
        g.fillRect(1, 1, width-1, height-1); 
        //设置画布的边框颜色 
        g.setColor(new Color(102,102,102));    
        // 绘制边框
        g.drawRect(0, 0, width-1, height-1);  
        //设置画布字体
        g.setFont(mFont);     
       // 设置随机线的颜色
        g.setColor(getRandColor(160,200));     
    
        //画随机线,使图象中的认证码不易被其它程序探测到。
        for (int i=0;i<155;i++)     
        {     
            int x = random.nextInt(width - 1);     
            int y = random.nextInt(height - 1);     
            int xl = random.nextInt(6) + 1;     
            int yl = random.nextInt(12) + 1;  
            // 在画布的指定位置画干扰线
            g.drawLine(x,y,x + xl,y + yl);     
        }     
    
        //从另一方向画随机线     
        for (int i = 0;i < 70;i++)     
        {     
            int x = random.nextInt(width - 1);     
            int y = random.nextInt(height - 1);     
            int xl = random.nextInt(12) + 1;     
            int yl = random.nextInt(6) + 1;     
            g.drawLine(x,y,x - xl,y - yl);     
        }     
    
        //生成随机数,并将随机数字转换为字母     
        String sRand="";     
        for (int i=0;i<6;i++)     
        {     
        	// 将随机生成的数字转化为英文字符
            int itmp = random.nextInt(26) + 65;     
            char ctmp = (char)itmp;     
            sRand += String.valueOf(ctmp);  
            // 设置每一个字母的颜色
            g.setColor(new Color(20+random.nextInt(110),20+random.nextInt(110),20+random.nextInt(110)));     
            // 字符前面预留的宽度 15*i+10,高度16
        	// 到这里就生成了一张有背景色,有边框,有指定个数的验证码被画在了指定的位置并且每个字的颜色不同。
            g.drawString(String.valueOf(ctmp),15*i+10,16);     
        }     
    
        HttpSession session = request.getSession(true);   
        // 添加到会话中
        session.setAttribute("rand",sRand);    
        ServletOutputStream outStream= response.getOutputStream();
        ImageIO.write(image, "JPEG",outStream );    
        //使用dispose()方法关闭窗体会释放该窗体的占用的部分资源
        g.dispose();   
        // 释放资源
        outStream.close();
        
    }     
}

3、汉字验证码:

package com.soufun;

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

import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * Servlet implementation class DrawImage
 */
<span style="color:#FF0000;">// 可以使用web.xml 配置servlet</span>
@WebServlet("/DrawImage")
public class DrawImage extends HttpServlet {
    public static final int WIDTH = 120;
    public static final int HEIGHT = 30;
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        this.doPost(request, response);
    }
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        request.setCharacterEncoding("utf-8");
        response.setContentType("text/html;charset=utf-8");
        // 创建缓存
        BufferedImage bi = new BufferedImage(WIDTH, HEIGHT,
                BufferedImage.TYPE_INT_RGB);
        // 获得画布
        Graphics g = bi.getGraphics();
        // 设置背影色
        setBackGround(g);
        // 设置边框
        setBorder(g);
        // 画干扰线
        drawRandomLine(g);
        // 写随机数
        String random = drawRandomNum((Graphics2D) g);
        // 将随机汉字存在session中
        request.getSession().setAttribute("checkcode", random);
        // 将图形写给浏览器
        response.setContentType("image/jpeg");
        // 发头控制浏览器不要缓存
        response.setDateHeader("expries", -1);
        response.setHeader("Cache-Control", "no-cache");
        response.setHeader("Pragma", "no-cache");
        // 将图片写给浏览器
        ServletOutputStream outStream=  response.getOutputStream();
        ImageIO.write(bi, "jpg", outStream);
        //使用dispose()方法关闭窗体会释放该窗体的占用的部分资源
        g.dispose();
        // 释放流资源
        outStream.close();
    }
    /**
     * 设置背景色
     * 
     * @param g
     */
    private void setBackGround(Graphics g) {
        // 设置颜色
        g.setColor(Color.WHITE);
        // 填充区域
        g.fillRect(0, 0, WIDTH, HEIGHT);
    }
    /**
     * 设置边框
     * 
     * @param g
     */
    private void setBorder(Graphics g) {
        // 设置边框颜色
        g.setColor(Color.BLUE);
        // 边框区域
        g.drawRect(1, 1, WIDTH - 2, HEIGHT - 2);
    }
    /**
     * 画随机线条
     * 
     * @param g
     */
    private void drawRandomLine(Graphics g) {
        // 设置颜色
         g.setColor(Color.GREEN);
        //g.setColor(Color.lightGray);
        // 设置线条个数并画线,使图象中的认证码不易被其它程序探测到。
        for (int i = 0; i < 10; i++) {
            int x1 = new Random().nextInt(WIDTH);
            int y1 = new Random().nextInt(HEIGHT);
            int x2 = new Random().nextInt(WIDTH);
            int y2 = new Random().nextInt(HEIGHT);
            g.drawLine(x1, y1, x2, y2);
        }
    }
    /**
     * 画随机汉字
     * 
     * @param g
     * @return
     */
    private String drawRandomNum(Graphics2D g) {
        StringBuffer sb = new StringBuffer();
        // 设置颜色
        g.setColor(Color.BLUE);
        // 设置字体
        g.setFont(new Font("宋体", Font.BOLD, 20));
        // 准备常用汉字集
        String base = "\u7684\u4e00\u4e86\u662f\u6211\u4e0d\u5728\u4eba\u4eec\u6709\u6765\u4ed6\u8fd9\u4e0a\u7740\u4e2a\u5730\u5230\u5927\u91cc\u8bf4\u5c31\u53bb\u5b50\u5f97\u4e5f\u548c\u90a3\u8981\u4e0b\u770b\u5929\u65f6\u8fc7\u51fa\u5c0f\u4e48\u8d77\u4f60\u90fd\u628a\u597d\u8fd8\u591a\u6ca1\u4e3a\u53c8\u53ef\u5bb6\u5b66\u53ea\u4ee5\u4e3b\u4f1a\u6837\u5e74\u60f3\u751f\u540c\u8001\u4e2d\u5341\u4ece\u81ea\u9762\u524d\u5934\u9053\u5b83\u540e\u7136\u8d70\u5f88\u50cf\u89c1\u4e24\u7528\u5979\u56fd\u52a8\u8fdb\u6210\u56de\u4ec0\u8fb9\u4f5c\u5bf9\u5f00\u800c\u5df1\u4e9b\u73b0\u5c71\u6c11\u5019\u7ecf\u53d1\u5de5\u5411\u4e8b\u547d\u7ed9\u957f\u6c34\u51e0\u4e49\u4e09\u58f0\u4e8e\u9ad8\u624b\u77e5\u7406\u773c\u5fd7\u70b9\u5fc3\u6218\u4e8c\u95ee\u4f46\u8eab\u65b9\u5b9e\u5403\u505a\u53eb\u5f53\u4f4f\u542c\u9769\u6253\u5462\u771f\u5168\u624d\u56db\u5df2\u6240\u654c\u4e4b\u6700\u5149\u4ea7\u60c5\u8def\u5206\u603b\u6761\u767d\u8bdd\u4e1c\u5e2d\u6b21\u4eb2\u5982\u88ab\u82b1\u53e3\u653e\u513f\u5e38\u6c14\u4e94\u7b2c\u4f7f\u5199\u519b\u5427\u6587\u8fd0\u518d\u679c\u600e\u5b9a\u8bb8\u5feb\u660e\u884c\u56e0\u522b\u98de\u5916\u6811\u7269\u6d3b\u90e8\u95e8\u65e0\u5f80\u8239\u671b\u65b0\u5e26\u961f\u5148\u529b\u5b8c\u5374\u7ad9\u4ee3\u5458\u673a\u66f4\u4e5d\u60a8\u6bcf\u98ce\u7ea7\u8ddf\u7b11\u554a\u5b69\u4e07\u5c11\u76f4\u610f\u591c\u6bd4\u9636\u8fde\u8f66\u91cd\u4fbf\u6597\u9a6c\u54ea\u5316\u592a\u6307\u53d8\u793e\u4f3c\u58eb\u8005\u5e72\u77f3\u6ee1\u65e5\u51b3\u767e\u539f\u62ff\u7fa4\u7a76\u5404\u516d\u672c\u601d\u89e3\u7acb\u6cb3\u6751\u516b\u96be\u65e9\u8bba\u5417\u6839\u5171\u8ba9\u76f8\u7814\u4eca\u5176\u4e66\u5750\u63a5\u5e94\u5173\u4fe1\u89c9\u6b65\u53cd\u5904\u8bb0\u5c06\u5343\u627e\u4e89\u9886\u6216\u5e08\u7ed3\u5757\u8dd1\u8c01\u8349\u8d8a\u5b57\u52a0\u811a\u7d27\u7231\u7b49\u4e60\u9635\u6015\u6708\u9752\u534a\u706b\u6cd5\u9898\u5efa\u8d76\u4f4d\u5531\u6d77\u4e03\u5973\u4efb\u4ef6\u611f\u51c6\u5f20\u56e2\u5c4b\u79bb\u8272\u8138\u7247\u79d1\u5012\u775b\u5229\u4e16\u521a\u4e14\u7531\u9001\u5207\u661f\u5bfc\u665a\u8868\u591f\u6574\u8ba4\u54cd\u96ea\u6d41\u672a\u573a\u8be5\u5e76\u5e95\u6df1\u523b\u5e73\u4f1f\u5fd9\u63d0\u786e\u8fd1\u4eae\u8f7b\u8bb2\u519c\u53e4\u9ed1\u544a\u754c\u62c9\u540d\u5440\u571f\u6e05\u9633\u7167\u529e\u53f2\u6539\u5386\u8f6c\u753b\u9020\u5634\u6b64\u6cbb\u5317\u5fc5\u670d\u96e8\u7a7f\u5185\u8bc6\u9a8c\u4f20\u4e1a\u83dc\u722c\u7761\u5174\u5f62\u91cf\u54b1\u89c2\u82e6\u4f53\u4f17\u901a\u51b2\u5408\u7834\u53cb\u5ea6\u672f\u996d\u516c\u65c1\u623f\u6781\u5357\u67aa\u8bfb\u6c99\u5c81\u7ebf\u91ce\u575a\u7a7a\u6536\u7b97\u81f3\u653f\u57ce\u52b3\u843d\u94b1\u7279\u56f4\u5f1f\u80dc\u6559\u70ed\u5c55\u5305\u6b4c\u7c7b\u6e10\u5f3a\u6570\u4e61\u547c\u6027\u97f3\u7b54\u54e5\u9645\u65e7\u795e\u5ea7\u7ae0\u5e2e\u5566\u53d7\u7cfb\u4ee4\u8df3\u975e\u4f55\u725b\u53d6\u5165\u5cb8\u6562\u6389\u5ffd\u79cd\u88c5\u9876\u6025\u6797\u505c\u606f\u53e5\u533a\u8863\u822c\u62a5\u53f6\u538b\u6162\u53d4\u80cc\u7ec6";
        int x = 5;
        // 控制字数
        for (int i = 0; i < 4; i++) {
            // 设置字体旋转角度
            int degree = new Random().nextInt() % 30;
            // 截取汉字
            String ch = base.charAt(new Random().nextInt(base.length())) + "";
            sb.append(ch);
            // 正向角度
            g.rotate(degree * Math.PI / 180, x, 20);
            g.drawString(ch, x, 20);
            // 反向角度
            g.rotate(-degree * Math.PI / 180, x, 20);
            x += 30;
        }
        return sb.toString();
    }
}


4、前端展示代码:

<%@ page language="java" contentType="text/html; charset=gbk"
    pageEncoding="gbk"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>

<meta http-equiv="Content-Type" content="text/html; charset=gbk">
<title>Insert title here</title>

<head>
<script type="text/javascript">  

function changeCode() {    

var imgNode = document.getElementById("vimg");                    
  
imgNode.src = "/Img/ImgVarificationCodeServlet?t=" + Math.random();  // 防止浏览器缓存的问题       

}      
</script>  
  </head>  
  <body>  
    <form action="checkServlet" method="post">  
        <label>输入验证码</label><br/>  
        <input type="text" name="randomCode"/><img id="vimg"  title="点击更换" οnclick="changeCode();" src="/Img/ImgVarificationCodeServlet"><br/>  
        <input type="submit" value="submit">  
    </form>  
  </body> 
</html>



二、生成的验证码存放在memcaches  或者 redis替换session

1、使用memcaches 或redis 来存放验证码好处就是无域名或者session限制,并且重启服务器不受影响

2、但是有个条件那就是每一个用户都一个唯一能确定的key ,这样存在memcaches中或redis中 才能在提交的时候做判断


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值