我們很多時候的開發(fā)場景都會用到驗證碼的地方雹顺,今天我學習的時候,看到的一個java實現(xiàn)驗證碼的功能廊遍,一個方法可以實現(xiàn)兩種驗證方式,一個就是計算的方式贩挣,另一個就是字符串方式喉前。如下圖所示:
這兩種方式只需要一個參數(shù)控制,就可以隨意選擇你說喜歡的驗證方式
接下來我們就進入編碼吧
第一步:添加pom依賴
<!--驗證碼 -->
<dependency>
<groupId>com.github.penggle</groupId>
<artifactId>kaptcha</artifactId>
<version>2.3.2</version>
</dependency>
第二步:驗證碼文本生成器
package com.chuxin.shiro.config;
import com.google.code.kaptcha.text.impl.DefaultTextCreator;
import java.security.SecureRandom;
import java.util.Random;
/**
* @FileName: KaptchaTextCreator
* @Description: 驗證碼文本生成器
* @author: myp
* @create: 2019-11-04 17:32
*/
public class KaptchaTextCreator extends DefaultTextCreator {
private static final String[] CNUMBERS = "0,1,2,3,4,5,6,7,8,9,10".split(",");
@Override
public String getText() {
Integer result = 0;
Random random = new SecureRandom();
int x = random.nextInt(10);
int y = random.nextInt(10);
StringBuilder suChinese = new StringBuilder();
int randomoperands = (int) Math.round(Math.random() * 2);
if (randomoperands == 0) {
result = x * y;
suChinese.append(CNUMBERS[x]);
suChinese.append("*");
suChinese.append(CNUMBERS[y]);
} else if (randomoperands == 1) {
if (!(x == 0) && y % x == 0) {
result = y / x;
suChinese.append(CNUMBERS[y]);
suChinese.append("/");
suChinese.append(CNUMBERS[x]);
} else {
result = x + y;
suChinese.append(CNUMBERS[x]);
suChinese.append("+");
suChinese.append(CNUMBERS[y]);
}
} else if (randomoperands == 2) {
if (x >= y) {
result = x - y;
suChinese.append(CNUMBERS[x]);
suChinese.append("-");
suChinese.append(CNUMBERS[y]);
} else {
result = y - x;
suChinese.append(CNUMBERS[y]);
suChinese.append("-");
suChinese.append(CNUMBERS[x]);
}
} else {
result = x + y;
suChinese.append(CNUMBERS[x]);
suChinese.append("+");
suChinese.append(CNUMBERS[y]);
}
suChinese.append("=?@" + result);
return suChinese.toString();
}
}
第三步:編寫配置類
package com.chuxin.shiro.config;
import com.google.code.kaptcha.impl.DefaultKaptcha;
import com.google.code.kaptcha.util.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import static com.google.code.kaptcha.Constants.*;
import java.util.Properties;
/**
* @FileName: CaptchaConfig
* @Description: 驗證碼配置
* @author: myp
* @create: 2019-11-04 17:21
*/
@Configuration
public class CaptchaConfig {
@Bean(name = "captchaProducer")
public DefaultKaptcha getKaptchaBean() {
DefaultKaptcha defaultKaptcha = new DefaultKaptcha();
Properties properties = new Properties();
// 是否有邊框 默認為true 我們可以自己設置yes王财,no
properties.setProperty(KAPTCHA_BORDER, "yes");
// 驗證碼文本字符顏色 默認為Color.BLACK
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "black");
// 驗證碼圖片寬度 默認為200
properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");
// 驗證碼圖片高度 默認為50
properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");
// 驗證碼文本字符大小 默認為40
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "38");
// KAPTCHA_SESSION_KEY
properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCode");
// 驗證碼文本字符長度 默認為5
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "4");
// 驗證碼文本字體樣式 默認為new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize)
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier");
// 圖片樣式 水紋com.google.code.kaptcha.impl.WaterRipple 魚眼com.google.code.kaptcha.impl.FishEyeGimpy 陰影com.google.code.kaptcha.impl.ShadowGimpy
properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy");
Config config = new Config(properties);
defaultKaptcha.setConfig(config);
return defaultKaptcha;
}
@Bean(name = "captchaProducerMath")
public DefaultKaptcha getKaptchaBeanMath() {
DefaultKaptcha defaultKaptcha = new DefaultKaptcha();
Properties properties = new Properties();
// 是否有邊框 默認為true 我們可以自己設置yes卵迂,no
properties.setProperty(KAPTCHA_BORDER, "yes");
// 邊框顏色 默認為Color.BLACK
properties.setProperty(KAPTCHA_BORDER_COLOR, "105,179,90");
// 驗證碼文本字符顏色 默認為Color.BLACK
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_COLOR, "blue");
// 驗證碼圖片寬度 默認為200
properties.setProperty(KAPTCHA_IMAGE_WIDTH, "160");
// 驗證碼圖片高度 默認為50
properties.setProperty(KAPTCHA_IMAGE_HEIGHT, "60");
// 驗證碼文本字符大小 默認為40
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_SIZE, "35");
// KAPTCHA_SESSION_KEY
properties.setProperty(KAPTCHA_SESSION_CONFIG_KEY, "kaptchaCodeMath");
// 驗證碼文本生成器
properties.setProperty(KAPTCHA_TEXTPRODUCER_IMPL, "com.chuxin.shiro.config.KaptchaTextCreator");
// 驗證碼文本字符間距 默認為2
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_SPACE, "3");
// 驗證碼文本字符長度 默認為5
properties.setProperty(KAPTCHA_TEXTPRODUCER_CHAR_LENGTH, "6");
// 驗證碼文本字體樣式 默認為new Font("Arial", 1, fontSize), new Font("Courier", 1, fontSize)
properties.setProperty(KAPTCHA_TEXTPRODUCER_FONT_NAMES, "Arial,Courier");
// 驗證碼噪點顏色 默認為Color.BLACK
properties.setProperty(KAPTCHA_NOISE_COLOR, "white");
// 干擾實現(xiàn)類
properties.setProperty(KAPTCHA_NOISE_IMPL, "com.google.code.kaptcha.impl.NoNoise");
// 圖片樣式 水紋com.google.code.kaptcha.impl.WaterRipple 魚眼com.google.code.kaptcha.impl.FishEyeGimpy 陰影com.google.code.kaptcha.impl.ShadowGimpy
properties.setProperty(KAPTCHA_OBSCURIFICATOR_IMPL, "com.google.code.kaptcha.impl.ShadowGimpy");
Config config = new Config(properties);
defaultKaptcha.setConfig(config);
return defaultKaptcha;
}
}
第四步:編寫我們的控制類
package com.chuxin.shiro.controller;
import com.google.code.kaptcha.Constants;
import com.google.code.kaptcha.Producer;
import io.swagger.annotations.Api;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.servlet.ModelAndView;
import javax.annotation.Resource;
import javax.imageio.ImageIO;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.awt.image.BufferedImage;
import java.io.IOException;
/**
* @FileName: CaptchaController
* @Description: 驗證碼控制類
* @author: myp
* @create: 2019-11-04 16:55
*/
@RestController
@RequestMapping("/captcha")
@Api(value = "驗證碼接口", description = "驗證碼接口", tags = {"驗證碼接口"})
public class CaptchaController {
@Resource(name = "captchaProducer")
private Producer captchaProducer;
@Resource(name = "captchaProducerMath")
private Producer captchaProducerMath;
@Value("${chuxin.captchaType}")
private String captchaType;
/**
* 驗證碼生成
*/
@GetMapping(value = "/captchaImage")
public ModelAndView getKaptchaImage(HttpServletRequest request, HttpServletResponse response) {
ServletOutputStream out = null;
try {
HttpSession session = request.getSession();
response.setDateHeader("Expires", 0);
response.setHeader("Cache-Control", "no-store, no-cache, must-revalidate");
response.addHeader("Cache-Control", "post-check=0, pre-check=0");
response.setHeader("Pragma", "no-cache");
response.setContentType("image/jpeg");
String capStr = null;
String code = null;
BufferedImage bi = null;
if ("math".equals(captchaType)) {
String capText = captchaProducerMath.createText();
capStr = capText.substring(0, capText.lastIndexOf("@"));
code = capText.substring(capText.lastIndexOf("@") + 1);
bi = captchaProducerMath.createImage(capStr);
} else if ("char".equals(captchaType)) {
capStr = code = captchaProducer.createText();
bi = captchaProducer.createImage(capStr);
}
session.setAttribute(Constants.KAPTCHA_SESSION_KEY, code);
out = response.getOutputStream();
ImageIO.write(bi, "jpg", out);
out.flush();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (out != null) {
out.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
}
最后一個步:在配置文件中配置你想選擇的驗證碼類型
# math:為四則運算的形式;char:為字符串形式(由數(shù)字和字母組成)
chuxin:
captchaType: math
這樣我們就完成了驗證碼的生成绒净,對于如何進行存儲见咒,如何進行校驗就很簡單了,使用緩存也好挂疆,使用數(shù)據(jù)庫也好改览,根據(jù)自己的情況而定就可以啦。