微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

java实现图片滑动验证(包含前端代码)

这篇文章主要为大家详细介绍了Java实现图片滑动验证,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

前言

1、下面是一个效果展示;

2、先抱怨一下,在博客上面的抄袭真的非常严重,为了实现一个图片滑动验证,我搜索了挺久的资料,不过内容翻来覆去就是同样的内容,千篇一律,作者还各不相同;内容相同我就不多说了,毕竟能解决问题就行,然而恰恰相反,这些东西都没有为我实质性地解决问题。可能图片验证一个需要前后台同时交互的功能吧,从业的人员大部分都是偏向后台或者偏向前台的,所以写出来的博客都不能完整阐述整个流程,下面是我自己实践完成的内容,记录一下,供各位参阅斧正。

注:由于使用到的控件和工具较多,有许多地方做了省略,这里只做核心流程的记录。

一、后端图片裁剪与生成

首先是一个图片处理工具VerifyImageUtil.class,它主要的作用是生成两张图片:一张被扣除了一部分的原始图片;一张抠出来图片。两两结合,可以组成一张完整的图片。原始图片(target目录)提供了20张,规格都是590*360的;抠图需要的模板图(template目录)有4张,规格都是93*360的(图片等各种资源会在文末给出)。将图片资源导入到我们项目的静态资源路径下(你也可以通过其他方式存储它们),我这边是Spring Boot的项目,所以就放在resource下的static目录下了:

下面是 VerifyImageUtil.class

package com.mine.risk.util; import org.apache.commons.lang.StringUtils; import javax.imageio.ImageIO; import javax.imageio.ImageReadParam; import javax.imageio.ImageReader; import javax.imageio.stream.ImageInputStream; import java.awt.*; import java.awt.image.BufferedImage; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.text.NumberFormat; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.Random; /** * 滑块验证工具类 * @author : spirit * @date : Created in 10:57 2019/9/05 */ public class VerifyImageUtil { /** 源文件宽度 */ private static final int ORI_WIDTH = 590; /** 源文件高度 */ private static final int ORI_HEIGHT = 360; /** 抠图坐标x */ private static int X; /** 抠图坐标y */ private static int Y; /** 模板图宽度 */ private static int WIDTH; /** 模板图高度 */ private static int HEIGHT; public static int getX() { return X; } public static int getY() { return Y; } /** * 根据模板切图 * @param templateFile 模板文件 * @param targetFile 目标文件 * @param templateType 模板文件类型 * @param targettype 目标文件类型 * @return 切图map集合 * @throws Exception 异常 */ public static Map pictureTemplatesCut(File templateFile, File targetFile, String templateType, String targettype) throws Exception { Map pictureMap = new HashMap(2); if (StringUtils.isEmpty(templateType) || StringUtils.isEmpty(targettype)) { throw new RuntimeException("file type is empty"); } InputStream targetIs = new FileInputStream(targetFile); // 模板图 BufferedImage imageTemplate = ImageIO.read(templateFile); WIDTH = imageTemplate.getWidth(); HEIGHT = imageTemplate.getHeight(); // 随机生成抠图坐标 generateCutoutCoordinates(); // 最终图像 BufferedImage newImage = new BufferedImage(WIDTH, HEIGHT, imageTemplate.getType()); Graphics2D graphics = newImage.createGraphics(); graphics.setBackground(Color.white); int bold = 5; // 获取感兴趣的目标区域 BufferedImage targetimageNoDeal = getTargetArea(X, Y, WIDTH, HEIGHT, targetIs, targettype); // 根据模板图片抠图 newImage = dealCutPictureByTemplate(targetimageNoDeal, imageTemplate, newImage); // 设置“抗锯齿”的属性 graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); graphics.setstroke(new Basicstroke(bold, Basicstroke.CAP_BUTT, Basicstroke.JOIN_BEVEL)); graphics.drawImage(newImage, 0, 0, null); graphics.dispose(); //新建流。 ByteArrayOutputStream os = new ByteArrayOutputStream(); //利用ImageIO类提供的write方法,将bi以png图片的数据模式写入流。 ImageIO.write(newImage, "png", os); byte[] newImages = os.toByteArray(); pictureMap.put("newImage", newImages); // 源图生成遮罩 BufferedImage oriImage = ImageIO.read(targetFile); byte[] oricopyImages = dealOriPictureByTemplate(oriImage, imageTemplate, X, Y); pictureMap.put("oricopyImage", oricopyImages); System.out.println("X="+X+";y="+Y); return pictureMap; } /** * 抠图后原图生成 * @param oriImage 原始图片 * @param templateImage 模板图片 * @param x 坐标X * @param y 坐标Y * @return 添加遮罩层后的原始图片 * @throws Exception 异常 */ private static byte[] dealOriPictureByTemplate(BufferedImage oriImage, BufferedImage templateImage, int x, int y) throws Exception { // 源文件备份图像矩阵 支持alpha通道的rgb图像 BufferedImage oricopyImage = new BufferedImage(oriImage.getWidth(), oriImage.getHeight(), BufferedImage.TYPE_4BYTE_ABGR); // 源文件图像矩阵 int[][] oriImageData = getData(oriImage); // 模板图像矩阵 int[][] templateImageData = getData(templateImage); //copy 源图做不透明处理 for (int i = 0; i > 8)); int b = (0xff & (rgb >> 16)); //无透明处理 rgb = r + (g > 8)); int b = (0xff & (rgb_ori >> 16)); rgb_ori = r + (g imageReaderList = ImageIO.getimageReadersByFormatName(fileType); ImageReader imageReader = imageReaderList.next(); // 获取图片流 ImageInputStream iis = ImageIO.createImageInputStream(ois); // 输入源中的图像将只按顺序读取 imageReader.setInput(iis, true); ImageReadParam param = imageReader.getDefaultReadparam(); Rectangle rec = new Rectangle(x, y, targetWidth, targetHeight); param.setSourceRegion(rec); return imageReader.read(0, param); } /** * 生成图像矩阵 * @param bufferedImage 图片流 * @return 图像矩阵 */ private static int[][] getData(BufferedImage bufferedImage){ int[][] data = new int[bufferedImage.getWidth()][bufferedImage.getHeight()]; for (int i = 0; i

有了工具类,就可以开始生成图片内容了,我这边直接在Spring的控制器生成内容并返回

@RequestMapping("createImgValidate") @ResponseBody public Message createImgValidate(SmsverificationCodeVo vo){ try { Integer templateNum = new Random().nextInt(4) + 1; Integer targetNum = new Random().nextInt(20) + 1; File templateFile = ResourceUtils.getFile("classpath:static/images/validate/template/"+templateNum+".png"); File targetFile = ResourceUtils.getFile("classpath:static/images/validate/target/"+targetNum+".jpg"); Map pictureMap = VerifyImageUtil.pictureTemplatesCut(templateFile, targetFile, ConstString.IMAGE_TYPE_PNG,ConstString.IMAGE_TYPE_JPG); // 将生成的偏移位置信息设置到redis中 String key = ConstString.WEB_VALID_IMAGE_PREFIX + vo.getTelephone(); boolean verified = redisUtil.exists(key); if(verified){ redisUtil.del(key); } redisUtil.set(key,(VerifyImageUtil.getX()+67)+"",SmsUtil.VALID_IMAGE_TIMEOUT); return ResponseUtil.success(pictureMap); } catch (Exception e) { e.printstacktrace(); return ResponseUtil.info(ResponseEnum.BUSInesS_ERROR); } }

基本的逻辑是从静态资源中随机加载一张target图片和一张template图片,放到图片处理工具中,处理并返回我们需要的两张图片生成图片以后,就可以直接返回这个Map了,它会以base64的方式返回到浏览器端。在这里,偏移的位置信息属于敏感内容,它会参与前台传入偏移量的对比校验,所以我这里存到了redis中,返回的内容也就是Map,只不过我用了一个自定义的返回辅助方法(有兴趣的人也可以找我要这些辅助工具)。

二、前端展示图片

首先还是需要在Spring Boot对应的控制器中,加入生成视图的代码(我做图片滑动验证主要为了在发送手机验证码之前做校验,所以有一个手机号的参数)。

/** * 跳转图片验证界面 * @return 图片验证界面 */ @RequestMapping("imgValidate") public String toImgValidate(ModelMap map, String telephone){ map.addAttribute("telephone",telephone); return "component/imageValidate"; }

之后便是我们的HTML页码代码:imageValidate.html

图片验证

 换一组

拖动滑块完成拼图

然后是对应的JS逻辑代码:imageValidate.js。之前说过后台返回的图片是转成base64了的,所以我们在生成图片的时候,直接在img标签的src内容前加入data:image/png;base64,即可,注意又一个英文逗号。

var left = 0; $(function(){ // 初始化图片验证码 initimageValidate(); /* 初始化按钮拖动事件 */ // 鼠标点击事件 $("#sliderInner").mousedown(function(){ // 鼠标移动事件 document.onmousemove = function(ev) { left = ev.clientX; if(left >= 67 && left

最后是css样式代码:imageValidate.css

body{ overflow: hidden; } #container{ width: 100%; } .fontDiv{ margin: 16px 0; } .dragFont{ font-size: 16px; color: dodgerblue; } .imageDiv{ width: 590px; height: 360px; margin: 20px auto 0 auto; position: relative; } .resultDiv{ margin: 10px 20px; } #validateImage{ border-radius: 4px; } #slideImage{ position: absolute; top: 5px; left: 0; } #sliderOuter{ width: 590px; height: 40px; margin: 12px auto; border-radius: 20px; Box-shadow: 0 0 10px 5px darkgrey; display: flex; align-items: center; justify-content: center; position: relative; } #dragDiv{ width: 100%; height: 40px; position: absolute; font-size: 16px; color: dodgerblue; text-align: center; line-height: 40px; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } #sliderInner{ width: 94px; height: 40px; border-radius: 20px; font-size: 2rem; background-color: #28a745; cursor: pointer; position: absolute; left: 0; } #sliderInner i{ position: relative; top: -2px; left: 36px; color: white; } .coverIcon{ width: 100%; height: 100%; position: absolute; top: 0; }

资源包下载:Java图片滑动验证

更多关于验证码的文章请点击查看:《java验证码》

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程之家。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐