真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

Java中SSM+Shiro系統(tǒng)登錄驗(yàn)證碼的實(shí)現(xiàn)方法

 先給大家展示下效果圖:

創(chuàng)新互聯(lián)建站主營上饒網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營網(wǎng)站建設(shè)方案,App定制開發(fā),上饒h5小程序制作搭建,上饒網(wǎng)站營銷推廣歡迎上饒等地區(qū)企業(yè)咨詢

Java中SSM+Shiro系統(tǒng)登錄驗(yàn)證碼的實(shí)現(xiàn)方法

1、驗(yàn)證碼生成類:

import java.util.Random;
import java.awt.image.BufferedImage;
import java.awt.Graphics;
import java.awt.Font;
import java.awt.Color;
/**
 * 驗(yàn)證碼生成器類,可生成數(shù)字、大寫、小寫字母及三者混合類型的驗(yàn)證碼。 支持自定義驗(yàn)證碼字符數(shù)量; 支持自定義驗(yàn)證碼圖片的大小; 支持自定義需排除的特殊字符;
 * 支持自定義干擾線的數(shù)量; 支持自定義驗(yàn)證碼圖文顏色
 */
public class ValidateCode {
 /**
  * 驗(yàn)證碼類型為僅數(shù)字 0~9
  */
 public static final int TYPE_NUM_ONLY = 0;
 /**
  * 驗(yàn)證碼類型為僅字母,即大寫、小寫字母混合
  */
 public static final int TYPE_LETTER_ONLY = 1;
 /**
  * 驗(yàn)證碼類型為數(shù)字、大寫字母、小寫字母混合
  */
 public static final int TYPE_ALL_MIXED = 2;
 /**
  * 驗(yàn)證碼類型為數(shù)字、大寫字母混合
  */
 public static final int TYPE_NUM_UPPER = 3;
 /**
  * 驗(yàn)證碼類型為數(shù)字、小寫字母混合
  */
 public static final int TYPE_NUM_LOWER = 4;
 /**
  * 驗(yàn)證碼類型為僅大寫字母
  */
 public static final int TYPE_UPPER_ONLY = 5;
 /**
  * 驗(yàn)證碼類型為僅小寫字母
  */
 public static final int TYPE_LOWER_ONLY = 6;
 private ValidateCode() {
 }
 /**
  * 生成驗(yàn)證碼字符串
  * 
  * @param type
  *   驗(yàn)證碼類型,參見本類的靜態(tài)屬性
  * @param length
  *   驗(yàn)證碼長度,大于0的整數(shù)
  * @param exChars
  *   需排除的特殊字符(僅對(duì)數(shù)字、字母混合型驗(yàn)證碼有效,無需排除則為null)
  * @return 驗(yàn)證碼字符串
  */
 public static String generateTextCode(int type, int length, String exChars) {
  if (length <= 0)
   return "";
  StringBuffer code = new StringBuffer();
  int i = 0;
  Random r = new Random();
  switch (type) {
  // 僅數(shù)字
  case TYPE_NUM_ONLY:
   while (i < length) {
    int t = r.nextInt(10);
    if (exChars == null || exChars.indexOf(t + "") < 0) {// 排除特殊字符
     code.append(t);
     i++;
    }
   }
   break;
  // 僅字母(即大寫字母、小寫字母混合)
  case TYPE_LETTER_ONLY:
   while (i < length) {
    int t = r.nextInt(123);
    if ((t >= 97 || (t >= 65 && t <= 90)) && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  // 數(shù)字、大寫字母、小寫字母混合
  case TYPE_ALL_MIXED:
   while (i < length) {
    int t = r.nextInt(123);
    if ((t >= 97 || (t >= 65 && t <= 90) || (t >= 48 && t <= 57))
      && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  // 數(shù)字、大寫字母混合
  case TYPE_NUM_UPPER:
   while (i < length) {
    int t = r.nextInt(91);
    if ((t >= 65 || (t >= 48 && t <= 57)) && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  // 數(shù)字、小寫字母混合
  case TYPE_NUM_LOWER:
   while (i < length) {
    int t = r.nextInt(123);
    if ((t >= 97 || (t >= 48 && t <= 57)) && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  // 僅大寫字母
  case TYPE_UPPER_ONLY:
   while (i < length) {
    int t = r.nextInt(91);
    if ((t >= 65) && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  // 僅小寫字母
  case TYPE_LOWER_ONLY:
   while (i < length) {
    int t = r.nextInt(123);
    if ((t >= 97) && (exChars == null || exChars.indexOf((char) t) < 0)) {
     code.append((char) t);
     i++;
    }
   }
   break;
  }
  return code.toString();
 }
 /**
  * 已有驗(yàn)證碼,生成驗(yàn)證碼圖片
  * 
  * @param textCode
  *   文本驗(yàn)證碼
  * @param width
  *   圖片寬度
  * @param height
  *   圖片高度
  * @param interLine
  *   圖片中干擾線的條數(shù)
  * @param randomLocation
  *   每個(gè)字符的高低位置是否隨機(jī)
  * @param backColor
  *   圖片顏色,若為null,則采用隨機(jī)顏色
  * @param foreColor
  *   字體顏色,若為null,則采用隨機(jī)顏色
  * @param lineColor
  *   干擾線顏色,若為null,則采用隨機(jī)顏色
  * @return 圖片緩存對(duì)象
  */
 public static BufferedImage generateImageCode(String textCode, int width, int height, int interLine,
   boolean randomLocation, Color backColor, Color foreColor, Color lineColor) {
  BufferedImage bim = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
  Graphics g = bim.getGraphics();
  // 畫背景圖
  g.setColor(backColor == null ? getRandomColor() : backColor);
  g.fillRect(0, 0, width, height);
  // 畫干擾線
  Random r = new Random();
  if (interLine > 0) {
   int x = 0, y = 0, x1 = width, y1 = 0;
   for (int i = 0; i < interLine; i++) {
    g.setColor(lineColor == null ? getRandomColor() : lineColor);
    y = r.nextInt(height);
    y1 = r.nextInt(height);
    g.drawLine(x, y, x1, y1);
   }
  }
  // 寫驗(yàn)證碼
  // g.setColor(getRandomColor());
  // g.setColor(isSimpleColor?Color.BLACK:Color.WHITE);
  // 字體大小為圖片高度的80%
  int fsize = (int) (height * 0.8);
  int fx = height - fsize;
  int fy = fsize;
  g.setFont(new Font("Default", Font.PLAIN, fsize));
  // 寫驗(yàn)證碼字符
  for (int i = 0; i < textCode.length(); i++) {
   fy = randomLocation ? (int) ((Math.random() * 0.3 + 0.6) * height) : fy;// 每個(gè)字符高低是否隨機(jī)
   g.setColor(foreColor == null ? getRandomColor() : foreColor);
   g.drawString(textCode.charAt(i) + "", fx, fy);
   fx += fsize * 0.9;
  }
  g.dispose();
  return bim;
 }
 /**
  * 生成圖片驗(yàn)證碼
  * 
  * @param type
  *   驗(yàn)證碼類型,參見本類的靜態(tài)屬性
  * @param length
  *   驗(yàn)證碼字符長度,大于0的整數(shù)
  * @param exChars
  *   需排除的特殊字符
  * @param width
  *   圖片寬度
  * @param height
  *   圖片高度
  * @param interLine
  *   圖片中干擾線的條數(shù)
  * @param randomLocation
  *   每個(gè)字符的高低位置是否隨機(jī)
  * @param backColor
  *   圖片顏色,若為null,則采用隨機(jī)顏色
  * @param foreColor
  *   字體顏色,若為null,則采用隨機(jī)顏色
  * @param lineColor
  *   干擾線顏色,若為null,則采用隨機(jī)顏色
  * @return 圖片緩存對(duì)象
  */
 public static BufferedImage generateImageCode(int type, int length, String exChars, int width, int height,
   int interLine, boolean randomLocation, Color backColor, Color foreColor, Color lineColor) {
  String textCode = generateTextCode(type, length, exChars);
  BufferedImage bim = generateImageCode(textCode, width, height, interLine, randomLocation, backColor, foreColor,
    lineColor);
  return bim;
 }
 /**
  * 產(chǎn)生隨機(jī)顏色
  * 
  * @return
  */
 private static Color getRandomColor() {
  Random r = new Random();
  Color c = new Color(r.nextInt(255), r.nextInt(255), r.nextInt(255));
  return c;
 }
}

2、Controller

 /**
  * 生成驗(yàn)證碼
  * @param request
  * @param response
  * @throws IOException
  * @ValidateCode.generateTextCode(驗(yàn)證碼字符類型,驗(yàn)證碼長度,需排除的特殊字符)
  * @ValidateCode.generateImageCode(文本驗(yàn)證碼,圖片寬度,圖片高度,干擾線的條數(shù),字符的高低位置是否隨機(jī),圖片顏色,字體顏色,干擾線顏色)
  */
 @RequestMapping(value = "validateCode")
 public void validateCode(HttpServletRequest request, HttpServletResponse response) throws IOException {
  response.setHeader("Cache-Control", "no-cache");
  String verifyCode = ValidateCode.generateTextCode(ValidateCode.TYPE_NUM_LOWER, 4, null);
  request.getSession().setAttribute("validateCode", verifyCode);
  response.setContentType("image/jpeg");
  BufferedImage bim = ValidateCode.generateImageCode(verifyCode, 90, 30, 5, true, Color.WHITE, Color.BLUE, null);
  ImageIO.write(bim, "JPEG", response.getOutputStream());
 }
 /**
  * 登錄請(qǐng)求
  * @param 
  */
 @RequestMapping(value = "login", method = RequestMethod.POST, produces = "text/html; charset=utf-8")
 public String login(HttpServletRequest request, HttpServletResponse response, UserEntity user) {
  //首先進(jìn)行驗(yàn)證碼驗(yàn)證
  Session session = SecurityUtils.getSubject().getSession();
  String code = (String) session.getAttribute("validateCode");
  String submitCode = WebUtils.getCleanParam(request, "validateCode");
  if (StringUtils.isEmpty(submitCode) || !StringUtils.equals(code,submitCode.toLowerCase())) {
   request.setAttribute("LOGIN_ERROR_CODE", LoginConstant.LOGIN_ERROR_CODE_100000);
   request.setAttribute("LOGIN_ERROR_MESSAGE", LoginConstant.LOGIN_ERROR_MESSAGE_VALIDATECODE);
   return "login";
  }
  // 想要得到 SecurityUtils.getSubject() 的對(duì)象..訪問地址必須跟shiro的攔截地址內(nèi).不然后會(huì)報(bào)空指針
  Subject sub = SecurityUtils.getSubject();
  // 用戶輸入的賬號(hào)和密碼,,存到UsernamePasswordToken對(duì)象中..然后由shiro內(nèi)部認(rèn)證對(duì)比,
  // 認(rèn)證執(zhí)行者交由ShiroDbRealm中doGetAuthenticationInfo處理
  // 當(dāng)以上認(rèn)證成功后會(huì)向下執(zhí)行,認(rèn)證失敗會(huì)拋出異常
  UsernamePasswordToken token = new UsernamePasswordToken(user.getAccountName(), user.getPassWord());
  try {
   sub.login(token);
  } catch (LockedAccountException lae) {
   token.clear();
   request.setAttribute("LOGIN_ERROR_CODE", LoginConstant.LOGIN_ERROR_CODE_100002);
   request.setAttribute("LOGIN_ERROR_MESSAGE", LoginConstant.LOGIN_ERROR_MESSAGE_SYSTEMERROR);
   return "login";
  } catch (ExcessiveAttemptsException e) {
   token.clear();
   request.setAttribute("LOGIN_ERROR_CODE", LoginConstant.LOGIN_ERROR_CODE_100003);
   request.setAttribute("LOGIN_ERROR_MESSAGE","賬號(hào):" + user.getUserName() + LoginConstant.LOGIN_ERROR_MESSAGE_MAXERROR);
   return "login";
  } catch (AuthenticationException e) {
   token.clear();
   request.setAttribute("LOGIN_ERROR_CODE", LoginConstant.LOGIN_ERROR_CODE_100001);
   request.setAttribute("LOGIN_ERROR_MESSAGE", LoginConstant.LOGIN_ERROR_MESSAGE_USERERROR);
   return "login";
  }
  return "redirect:/index.shtml";
 }

注意:

登錄方法里面一些參數(shù)的定義:

public interface LoginConstant
{
 String LOGIN_ERROR_CODE_100000 = "100000";
 String LOGIN_ERROR_MESSAGE_VALIDATECODE = "驗(yàn)證碼輸入錯(cuò)誤,請(qǐng)重新輸入!";
 String LOGIN_ERROR_CODE_100001 = "100001";
 String LOGIN_ERROR_MESSAGE_USERERROR = "賬號(hào)或密碼錯(cuò)誤,請(qǐng)重新輸入!";
 String LOGIN_ERROR_CODE_100002 = "100002";
 String LOGIN_ERROR_MESSAGE_SYSTEMERROR = "用戶已經(jīng)被鎖定不能登錄,請(qǐng)與管理員聯(lián)系!";
 String LOGIN_ERROR_CODE_100003 = "100003";
 String LOGIN_ERROR_MESSAGE_MAXERROR = "登錄失敗次數(shù)過多,鎖定10分鐘!";
 String LOGIN_ERROR_CODE_100004 = "100004";
 String LOGIN_ERROR_MESSAGE_FORCELOGOUT = "您已經(jīng)被管理員強(qiáng)制退出,請(qǐng)重新登錄";
}

3、登錄jsp(重要代碼)

路徑信息:

<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path;
%>

js:用于更換驗(yàn)證碼圖片

 

登錄表單里面的標(biāo)簽:

  看不清?

4、Shiro匿名訪問配置(不配置無法生成驗(yàn)證碼圖片)


 
  
   
    /validateCode.shtml = anon//添加這行
   
  
 

以上所述是小編給大家介紹的Java中SSM+Shiro系統(tǒng)登錄驗(yàn)證碼的實(shí)現(xiàn)方法,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)創(chuàng)新互聯(lián)網(wǎng)站的支持!


當(dāng)前題目:Java中SSM+Shiro系統(tǒng)登錄驗(yàn)證碼的實(shí)現(xiàn)方法
標(biāo)題網(wǎng)址:http://weahome.cn/article/gssiod.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部