我是靠谱客的博主 积极银耳汤,这篇文章主要介绍java-通过类实现验证码(一),现在分享给大家,希望可以做个参考。

[java]  view plain copy print ?
  1. package com.jadyer.util;  
  2.   
  3. import java.awt.Color;  
  4. import java.awt.Font;  
  5. import java.awt.Graphics;  
  6. import java.awt.image.BufferedImage;  
  7. import java.io.File;  
  8. import java.io.IOException;  
  9. import java.util.Random;  
  10.   
  11. import javax.imageio.ImageIO;  
  12.   
  13. /** 
  14.  * 验证码生成器 
  15.  * @see -------------------------------------------------------------------------------------------------------------- 
  16.  * @see 可生成数字、大写、小写字母及三者混合类型的验证码 
  17.  * @see 支持自定义验证码字符数量,支持自定义验证码图片的大小,支持自定义需排除的特殊字符,支持自定义干扰线的数量,支持自定义验证码图文颜色 
  18.  * @see -------------------------------------------------------------------------------------------------------------- 
  19.  * @see 另外,给Shiro加入验证码有多种方式,也可以通过继承修改FormAuthenticationFilter类,通过Shiro去验证验证码 
  20.  * @see 而这里既然使用了SpringMVC,也为了简化操作,就使用此工具生成验证码,并在Controller中处理验证码的校验 
  21.  * @see -------------------------------------------------------------------------------------------------------------- 
  22.  * @create Sep 29, 2013 4:23:13 PM 
  23.  * @author 玄玉<http://blog.csdn.net/jadyer> 
  24.  */  
  25. public class VerifyCodeUtil {  
  26.     /** 
  27.      * 验证码类型为仅数字,即0~9 
  28.      */  
  29.     public static final int TYPE_NUM_ONLY = 0;  
  30.   
  31.     /** 
  32.      * 验证码类型为仅字母,即大小写字母混合 
  33.      */  
  34.     public static final int TYPE_LETTER_ONLY = 1;  
  35.   
  36.     /** 
  37.      * 验证码类型为数字和大小写字母混合 
  38.      */  
  39.     public static final int TYPE_ALL_MIXED = 2;  
  40.   
  41.     /** 
  42.      * 验证码类型为数字和大写字母混合 
  43.      */  
  44.     public static final int TYPE_NUM_UPPER = 3;  
  45.   
  46.     /** 
  47.      * 验证码类型为数字和小写字母混合 
  48.      */  
  49.     public static final int TYPE_NUM_LOWER = 4;  
  50.   
  51.     /** 
  52.      * 验证码类型为仅大写字母 
  53.      */  
  54.     public static final int TYPE_UPPER_ONLY = 5;  
  55.   
  56.     /** 
  57.      * 验证码类型为仅小写字母 
  58.      */  
  59.     public static final int TYPE_LOWER_ONLY = 6;  
  60.   
  61.     private VerifyCodeUtil(){}  
  62.       
  63.     /** 
  64.      * 生成随机颜色 
  65.      */  
  66.     private static Color generateRandomColor() {  
  67.         Random random = new Random();  
  68.         return new Color(random.nextInt(255), random.nextInt(255), random.nextInt(255));  
  69.     }  
  70.       
  71.       
  72.     /** 
  73.      * 生成图片验证码 
  74.      * @param type           验证码类型,参见本类的静态属性 
  75.      * @param length         验证码字符长度,要求大于0的整数 
  76.      * @param excludeString  需排除的特殊字符 
  77.      * @param width          图片宽度(注意此宽度若过小,容易造成验证码文本显示不全,如4个字符的文本可使用85到90的宽度) 
  78.      * @param height         图片高度 
  79.      * @param interLine      图片中干扰线的条数 
  80.      * @param randomLocation 每个字符的高低位置是否随机 
  81.      * @param backColor      图片颜色,若为null则表示采用随机颜色 
  82.      * @param foreColor      字体颜色,若为null则表示采用随机颜色 
  83.      * @param lineColor      干扰线颜色,若为null则表示采用随机颜色 
  84.      * @return 图片缓存对象 
  85.      */  
  86.     public static BufferedImage generateImageCode(int type, int length, String excludeString, int width, int height, int interLine, boolean randomLocation, Color backColor, Color foreColor, Color lineColor){  
  87.         String textCode = generateTextCode(type, length, excludeString);  
  88.         return generateImageCode(textCode, width, height, interLine, randomLocation, backColor, foreColor, lineColor);  
  89.     }  
  90.       
  91.   
  92.     /** 
  93.      * 生成验证码字符串 
  94.      * @param type          验证码类型,参见本类的静态属性 
  95.      * @param length        验证码长度,要求大于0的整数 
  96.      * @param excludeString 需排除的特殊字符(无需排除则为null) 
  97.      * @return 验证码字符串 
  98.      */  
  99.     public static String generateTextCode(int type, int length, String excludeString){  
  100.         if(length <= 0){  
  101.             return "";  
  102.         }  
  103.         StringBuffer verifyCode = new StringBuffer();  
  104.         int i = 0;  
  105.         Random random = new Random();  
  106.         switch(type){  
  107.             case TYPE_NUM_ONLY:  
  108.                 while(i < length){  
  109.                     int t = random.nextInt(10);  
  110.                     //排除特殊字符  
  111.                     if(null==excludeString || excludeString.indexOf(t+"")<0) {  
  112.                         verifyCode.append(t);  
  113.                         i++;  
  114.                     }  
  115.                 }  
  116.             break;  
  117.             case TYPE_LETTER_ONLY:  
  118.                 while(i < length){  
  119.                     int t = random.nextInt(123);  
  120.                     if((t>=97 || (t>=65&&t<=90)) && (null==excludeString||excludeString.indexOf((char)t)<0)){  
  121.                         verifyCode.append((char)t);  
  122.                         i++;  
  123.                     }  
  124.                 }  
  125.             break;  
  126.             case TYPE_ALL_MIXED:  
  127.                 while(i < length){  
  128.                     int t = random.nextInt(123);  
  129.                     if((t>=97 || (t>=65&&t<=90) || (t>=48&&t<=57)) && (null==excludeString||excludeString.indexOf((char)t)<0)){  
  130.                         verifyCode.append((char)t);  
  131.                         i++;  
  132.                     }  
  133.                 }  
  134.             break;  
  135.             case TYPE_NUM_UPPER:  
  136.                 while(i < length){  
  137.                     int t = random.nextInt(91);  
  138.                     if((t>=65 || (t>=48&&t<=57)) && (null==excludeString || excludeString.indexOf((char)t)<0)){  
  139.                         verifyCode.append((char)t);  
  140.                         i++;  
  141.                     }  
  142.                 }  
  143.             break;  
  144.             case TYPE_NUM_LOWER:  
  145.                 while(i < length){  
  146.                     int t = random.nextInt(123);  
  147.                     if((t>=97 || (t>=48&&t<=57)) && (null==excludeString || excludeString.indexOf((char)t)<0)){  
  148.                         verifyCode.append((char)t);  
  149.                         i++;  
  150.                     }  
  151.                 }  
  152.             break;  
  153.             case TYPE_UPPER_ONLY:  
  154.                 while(i < length){  
  155.                     int t = random.nextInt(91);  
  156.                     if((t >= 65) && (null==excludeString||excludeString.indexOf((char)t)<0)){  
  157.                         verifyCode.append((char)t);  
  158.                         i++;  
  159.                     }  
  160.                 }  
  161.             break;  
  162.             case TYPE_LOWER_ONLY:  
  163.                 while(i < length){  
  164.                     int t = random.nextInt(123);  
  165.                     if((t>=97) && (null==excludeString||excludeString.indexOf((char)t)<0)){  
  166.                         verifyCode.append((char)t);  
  167.                         i++;  
  168.                     }  
  169.                 }  
  170.             break;  
  171.         }  
  172.         return verifyCode.toString();  
  173.     }  
  174.   
  175.     /** 
  176.      * 已有验证码,生成验证码图片 
  177.      * @param textCode       文本验证码 
  178.      * @param width          图片宽度(注意此宽度若过小,容易造成验证码文本显示不全,如4个字符的文本可使用85到90的宽度) 
  179.      * @param height         图片高度 
  180.      * @param interLine      图片中干扰线的条数 
  181.      * @param randomLocation 每个字符的高低位置是否随机 
  182.      * @param backColor      图片颜色,若为null则表示采用随机颜色 
  183.      * @param foreColor      字体颜色,若为null则表示采用随机颜色 
  184.      * @param lineColor      干扰线颜色,若为null则表示采用随机颜色 
  185.      * @return 图片缓存对象 
  186.      */  
  187.     public static BufferedImage generateImageCode(String textCode, int width, int height, int interLine, boolean randomLocation, Color backColor, Color foreColor, Color lineColor){  
  188.         //创建内存图像  
  189.         BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);  
  190.         //获取图形上下文  
  191.         Graphics graphics = bufferedImage.getGraphics();  
  192.         //画背景图  
  193.         graphics.setColor(null==backColor ? generateRandomColor() : backColor);  
  194.         graphics.fillRect(00, width, height);  
  195.         //画干扰线  
  196.         Random random = new Random();  
  197.         if(interLine > 0){  
  198.             int x = 0, y = 0, x1 = width, y1 = 0;  
  199.             for(int i=0; i<interLine; i++){  
  200.                 graphics.setColor(null==lineColor ? generateRandomColor() : lineColor);  
  201.                 y = random.nextInt(height);  
  202.                 y1 = random.nextInt(height);  
  203.                 graphics.drawLine(x, y, x1, y1);  
  204.             }  
  205.         }  
  206.         //字体大小为图片高度的80%  
  207.         int fsize = (int)(height * 0.8);  
  208.         int fx = height - fsize;  
  209.         int fy = fsize;  
  210.         //设定字体  
  211.         graphics.setFont(new Font("Default", Font.PLAIN, fsize));  
  212.         //写验证码字符  
  213.         for(int i=0; i<textCode.length(); i++){  
  214.             fy = randomLocation ? (int)((Math.random()*0.3+0.6)*height) : fy;  
  215.             graphics.setColor(null==foreColor ? generateRandomColor() : foreColor);  
  216.             //将验证码字符显示到图象中  
  217.             graphics.drawString(textCode.charAt(i)+"", fx, fy);  
  218.             fx += fsize * 0.9;  
  219.         }  
  220.         graphics.dispose();  
  221.         return bufferedImage;  
  222.     }  
  223. }  

Controller类中主要方法:

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
<span style="font-size:12px;">/** * 获取验证码图片和文本(验证码文本会保存在HttpSession中) */ @RequestMapping("/getVerifyCodeImage") public void getVerifyCodeImage(HttpServletRequest request, HttpServletResponse response) throws IOException { //设置页面不缓存 response.setHeader("Pragma", "no-cache"); response.setHeader("Cache-Control", "no-cache"); response.setDateHeader("Expires", 0); String verifyCode = VerifyCodeUtil.generateTextCode(VerifyCodeUtil.TYPE_ALL_MIXED, 4, null); //将验证码放到HttpSession里面 request.getSession().setAttribute("verifyCode", verifyCode); System.out.println("本次生成的验证码为[" + verifyCode + "],已存放到HttpSession中"); //设置输出的内容的类型为JPEG图像 response.setContentType("image/jpeg"); BufferedImage bufferedImage = VerifyCodeUtil.generateImageCode(verifyCode, 90, 25, 4, true, Color.WHITE, Color.BLACK, null); //写给浏览器 ImageIO.write(bufferedImage, "JPEG", response.getOutputStream()); } /** * @param request * @return * 登录验证 */ @RequestMapping("/toLogin") public String toLogin(HttpServletRequest request) { String username = request.getParameter("username"); String password = request.getParameter("password"); //返回地址 String returnUrl="/login"; //User user =userService.queryUserByUserName(username); //获取HttpSession验证码 String verifyCode =(String) request.getSession().getAttribute("verifyCode"); //获取用户输入的验证码 String submitCode = WebUtils.getCleanParam(request,"verifyCode"); System.out.println("用户输入的验证码是:"+submitCode+";系统生成的验证码是:"+verifyCode); if(StringUtils.isEmpty(submitCode)|| !StringUtils.equalsIgnoreCase(verifyCode, submitCode)) { request.setAttribute("login_msg","验证码错误" ); return returnUrl; } //根据获取的用户名和密码封装成Token UsernamePasswordToken token =new UsernamePasswordToken(username,password); //是否记住用户 token.setRememberMe(true); //获取当前的subject Subject subject =SecurityUtils.getSubject(); try { System.out.println("对用户:["+username+"]进行登录验证,验证开始..."); //在调用了login方法后,SecurityManager会收到AuthenticationToken,并将其发送给已配置的Realm执行必须的认证检查 //每个Realm都能在必要时对提交的AuthenticationTokens作出反应 //所以这一步在调用login(token)方法时,它会走到MyRealm.doGetAuthenticationInfo()方法中,具体验证方式详见此方法 subject.login(token); System.out.println("对用户:["+username+"]进行登录验证,验证通过!"); returnUrl="/main"; }catch (UnknownAccountException e) { System.out.println("对用户:["+username+"]进行登录验证,验证未通过!错误:未知账号"); request.setAttribute("login_msg","未知账号"); }catch (IncorrectCredentialsException e) { System.out.println("对用户:["+username+"]进行登录验证,验证未通过!错误:密码错误"); request.setAttribute("login_msg", "密码错误"); }catch (LockedAccountException e) { System.out.println("对用户:["+username+"]进行登录验证,验证未通过!错误:账号被锁定"); request.setAttribute("login_msg", "账号被锁定"); }catch (ExcessiveAttemptsException e) { System.out.println("对用户:["+username+"]进行登录验证,验证未通过!错误:错误次数过多"); request.setAttribute("login_msg", "密码或用户名输入错误次数过多"); }catch (AuthenticationException e) { System.out.println("对用户:["+username+"]进行登录验证,验证未通过!错误:用户名或密码不正确"); request.setAttribute("login_msg", "身份认证失败,用户名或密码不正确"); } //验证是否登录成功 if(subject.isAuthenticated()) { System.out.println("用户:["+username+"]进行登录验证通过"); }else { token.clear(); } return returnUrl; }</span>


jsp页面 login.jsp

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <%@page isELIgnored="false"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; request.setAttribute("home", path); %> <!DOCTYPE HTML> <html lang="en-US"> <!-- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> --> <script type="text/javascript" src="<%=request.getContextPath()%>/res/login/prefixfree.min.js"></script> <script type="text/javascript" src="<%=request.getContextPath()%>/res/js/jquery-1.11.3.min.js"></script> <head> <meta charset="UTF-8"> <title>用户登录</title> <link rel="stylesheet" href="<%=request.getContextPath()%>/res/login/login.css" type="text/css"></link> <script type="text/javascript"> var home ="${home}"; var msg ="${login_msg }"; $(function(){ //生成验证码 $('#verifyCodeImage').click(function () { $(this).hide().attr('src', '<%=path%>/member/getVerifyCodeImage?' + Math.floor(Math.random()*100) ).fadeIn(); }); }); window.onbeforeunload = function(){ //关闭窗口时自动退出 if(event.clientX>360&&event.clientY<0||event.altKey){ alert(parent.document.location); } }; function changeCode() { //刷新 $('#verifyCodeImage').hide().attr('src', '<%=path%>/member/getVerifyCodeImage?' + Math.floor(Math.random()*100) ).fadeIn(); event.cancelBubble=true; } if(msg!="") { alert(msg); } </script> </head> <body> <div class="content"> <form action="<%=request.getContextPath()%>/member/toLogin" method="post" class="login-form"> <div class="username"> <input type="text" name="username" placeholder="emma.watson@gmail.com" autocomplete="on" /> <div id="loginMsg"></div> <span class="user-icon icon">u</span> </div> <div class="password"> <input type="password" name="password" placeholder="*******" /> <span class="password-icon icon">p</span> </div> <div class="code-div"> <input type="text" name="verifyCode" placeholder="请输入验证码" /> <img id="verifyCodeImage" src="<%=request.getContextPath()%>/member/getVerifyCodeImage"/> <!-- <a href="javascript:void(0)" οnclick="changeCode()">看不清?换一张</a> --> </div> <div class="account-control"> <input type="checkbox" name="rememberMe" id="Remember me" value="Remember me" checked="checked" /> <label for="Remember me" data-on="c" class="check"></label> <label for="Remember me" class="info">Remember me</label> <!-- <input type="hidden" name="rememberMe" value="true"> --> <button type="submit">Login</button> </div> <p class="not-registered">Not a registered user yet?<a>Sign up now!</a></p> </form> </div> </body> </html>

效果图:


来源:http://blog.csdn.net/jadyer/article/details/12185725


main方法小测试一下

[java]  view plain copy print ?
  1. public static void main(String[] args) throws IOException {  
  2.     String verifyCode = generateTextCode(TYPE_NUM_ONLY, 4null);  
  3.     System.out.println("verifyCode=" + verifyCode);  
  4.     BufferedImage bufferedImage = generateImageCode(verifyCode, 903010true, Color.WHITE, Color.BLACK, null);  
  5.     ImageIO.write(bufferedImage, "JPEG"new File("C:/Users/Jadyer/Desktop/aa.jpg"));  
  6. }  

最后

以上就是积极银耳汤最近收集整理的关于java-通过类实现验证码(一)的全部内容,更多相关java-通过类实现验证码(一)内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(58)

评论列表共有 0 条评论

立即
投稿
返回
顶部