失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > springmvc使用谷歌captcha生成图片验证码 并将验证码图片以二进制流的方式返回给前端

springmvc使用谷歌captcha生成图片验证码 并将验证码图片以二进制流的方式返回给前端

时间:2020-11-20 16:40:18

相关推荐

springmvc使用谷歌captcha生成图片验证码 并将验证码图片以二进制流的方式返回给前端

近期对登录注册与获取短信验证码的接口做了安全限制,其中一部分就用到了谷歌的captcha验证码,比如当用户连续三次登陆失败,那么之后的登录请求就需要用户输入谷歌的图形验证码。由于web端和app端调用的都是同一个获取谷歌验证码的接口,所以后端这个生成图片验证码的接口就与传统的直接将验证码图片写入到前端的方式不同,此时接口返回的应该这个经过base64编码后的验证码图片二进制流,然后由前端自行解析(web和app解析这个图片流的方式有些不一样)显示出来。主要代码可以借鉴这篇/4925054/2103196,下面分享我借鉴完后,自己做的一些修改。

首先是依赖的jar:

<!-- 谷歌验证码 --><dependency><groupId>com.github.penggle</groupId><artifactId>kaptcha</artifactId><version>2.3.2</version></dependency>

然后将生成验证码的com.google.code.kaptcha.impl.DefaultKaptcha类交给spring进行管理,在spring-mvc.xml文件中的配置是:

<!-- 谷歌验证码 --><bean id="captchaProducer" class="com.google.code.kaptcha.impl.DefaultKaptcha"><property name="config"><bean class="com.google.code.kaptcha.util.Config"><constructor-arg><props><prop key="kaptcha.border">yes</prop><prop key="kaptcha.border.color">105,179,90</prop><prop key="kaptcha.textproducer.font.color">blue</prop><prop key="kaptcha.image.width">200</prop><prop key="kaptcha.image.height">80</prop><prop key="kaptcha.textproducer.font.size">70</prop><prop key="kaptcha.session.key">code</prop><prop key="kaptcha.textproducer.char.length">4</prop><prop key="kaptcha.textproducer.font.names">宋体,楷体,微软雅黑</prop></props></constructor-arg></bean></property></bean>

具体的参数配置可以参考这篇博客:/blog/1308756,我也是直接拿人家的来用的!!!

接下来就是大家最喜欢的controller层的接口了:

package demo;import java.awt.image.BufferedImage;import java.io.ByteArrayOutputStream;import java.util.Map;import javax.imageio.ImageIO;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpSession;import mons.lang.StringUtils;import org.apache.log4j.Logger;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.RequestParam;import org.springframework.web.bind.annotation.ResponseBody;import com.google.code.kaptcha.Producer;import sun.misc.BASE64Encoder;import sy.util.redis.RedisUtils;/*** @author hqq* */@Controller@RequestMapping("/demo")public class RegAndLoginController {private static final Logger logger = Logger.getLogger(RegAndLoginController.class);@Autowiredprivate Producer captchaProducer;/*** 获取谷歌验证码* * @param request* @param response* @throws Exception*/@RequestMapping(value = "/captcha")@ResponseBodypublic Result getKaptchaImage(@RequestParam("mobile") String mobile, @RequestParam("type") String type,HttpServletRequest request) {Result result = new Result();if (StringUtils.isBlank(mobile) || StringUtils.isBlank(type)) {result.setSuccess(false);result.setMsg("非法请求,不予处理");return result;}//由于该接口是没有做权限校验的,任何人不登录都可以随意调用,所以此处建议做个每日访问次数的限制String ip = IpUtil.getIpAddr(request);String visitKey = ip +"_captcha_"+mobile;String visitCount = RedisUtils.get(visitKey);int count = 1;if (visitCount != null) {count = new Integer(visitCount);if (count >= 100) {result.setSuccess(true);result.setMsg("您今日调用此接口次数已达上限");return result;}count++;}RedisUtils.set(visitKey, count+"", DateUtils.getSurplusTime());if(!",login,register,".contains(","+type+",")){result.setSuccess(false);result.setMsg("非法请求,传入的参数错误");return result; }// 生成验证码String capText = captchaProducer.createText();String key=type+"_captcha_"+mobile;//当前验证码的时间可以设置的稍长些,防止用户在登录界面停留太长时间导致验证码失效了RedisUtils.set(key, capText.toLowerCase(), 60 * 30);//半个小时// 向客户端写出try {BufferedImage bi = captchaProducer.createImage(capText);// 生成图片验证码ByteArrayOutputStream outputStream = new ByteArrayOutputStream();ImageIO.write(bi, "jpg", outputStream);//高能预警!!!// 对字节数组Base64编码BASE64Encoder encoder = new BASE64Encoder();String img = encoder.encode(outputStream.toByteArray());result.setSuccess(true);result.setMsg("验证码生成成功");result.setImg(img);//返回图片的路径} catch (Exception e) {result.setSuccess(false);result.setMsg("系统异常:" + e.getMessage());}return result;}}

RedisUtils.java类请参照我的另一篇博客:/weixin_4666/article/details/89287418

工具类IpUtil.java如下:

package util;import .InetAddress;import javax.servlet.http.HttpServletRequest;public class IpUtil {/*** 获取请求的客户端的ip地址* * @param request* @return*/public static String getIpAddr(HttpServletRequest request) {String ipAddress = request.getHeader("x-forwarded-for");if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {ipAddress = request.getHeader("Proxy-Client-IP");}if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {ipAddress = request.getHeader("WL-Proxy-Client-IP");}if (ipAddress == null || ipAddress.length() == 0 || "unknown".equalsIgnoreCase(ipAddress)) {ipAddress = request.getRemoteAddr();if (ipAddress.equals("127.0.0.1") || ipAddress.equals("0:0:0:0:0:0:0:1")) {// 根据网卡取本机配置的IPInetAddress inet = null;try {inet = InetAddress.getLocalHost();ipAddress = inet.getHostAddress();} catch (Exception e) {e.printStackTrace();}}}// 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割if (ipAddress != null && ipAddress.length() > 15 && ipAddress.indexOf(",") > 0 ) { // "***.***.***.***".length()ipAddress = ipAddress.substring(0, ipAddress.indexOf(","));}return ipAddress;}}

web前端的js代码如下:

//获取图形验证码function getCodeImg(num,mobile) {$.ajax({type:"post",url:root+"/demo/captcha.action",async : false,data:{id:id,type:1},error:function(err){},success:function(r){if(r.success){$("#codeImg").attr('src','data:image/jpeg;base64,'+r.img+'');}else{pophint(r.msg)}}})}

效果如下:

app端的调用也是发ajax请求,具体代码我没有,反正是成功显示了,另外附上一篇博客,ios开发可以试试从这里找找:/sjl_leaf/article/details/48179299

为了防止该博文失效,截取一部分核心内容:

这是他的核心代码,安全起见,此处备份一下:

//参数在这里 + (void) setImageView:(UIImageView *)imageView WithString:(NSString *)avatar//二进制显示/*服务器返回:例如data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAADIAAAAyCAIAAACRXR/mAAAABnRSTlMAAAAAAABupgeRAAAAhElEQVRYhe3Y0QmAIBhF4YwGaZRGaNRGaJRGaYEyTj/RfTjnVYQPBRXbuh9DXuPfgOtkkWSRZJFkkUJZU3H+tsx3Q5VrLXS1ZJFkkWSRZJFkkWSRZJFkkUJZLfPH5vktX3mtv54buomySLJIskihrNBTvvWHP0V37oDQTZRFkkWSRQplnQDfEjVVBe3tAAAAAElFTkSuQmCC*/NSArray *imageArray = [avatar componentsSeparatedByString:@","];NSData *imageData = [[NSData alloc] initWithBase64EncodedString:imageArray[1] options:NSDataBase64DecodingIgnoreUnknownCharacters];imageView.image = [UIImage imageWithData:imageData];

至此,分享结束,希望能帮到各位!验证码的话,话说腾讯出的防水墙功能好像很牛的样子,大家感受一下:/online.html

springmvc使用谷歌captcha生成图片验证码 并将验证码图片以二进制流的方式返回给前端(app和pc端都能调用)

如果觉得《springmvc使用谷歌captcha生成图片验证码 并将验证码图片以二进制流的方式返回给前端》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。