JAVA实现图片验证码 超详细教程 您所在的位置:网站首页 java实现手机短信验证全过程图片 JAVA实现图片验证码 超详细教程

JAVA实现图片验证码 超详细教程

#JAVA实现图片验证码 超详细教程| 来源: 网络整理| 查看: 265

前言:

不知道大家有没有发现,我们在进行登录或者注册时,经常会遇到需要发送手机验证码的场景,基本上都是1到3分钟才允许点击一次获取验证码的按钮,这样做是为了限制用户频繁点击 「发送手机验证码」 按钮,毕竟平台发一条手机短信也是需要成本的。但是这样只是限制了点击,并不能限制暴力请求的情况。 这个时候图片验证码出现了,在你点击 「发送手机验证码」 按钮时需要正确输入图片中展示的验证码。 在密码输入错误之后再次登录时,都会有这样的图片验证码,这样做是为了防止用户利用机器人自动注册、登录、灌水。

实现: 本文需要依赖redis做缓存操作,如果项目中没有集成redis的话可以先添加依赖,pom文件中引入redis org.springframework.boot spring-boot-starter-data-redis 同时我们添加redis操作类 package com.blog.utils; import com.alibaba.fastjson.support.spring.FastJsonRedisSerializer; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.HashOperations; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.core.ValueOperations; import org.springframework.data.redis.serializer.StringRedisSerializer; import org.springframework.stereotype.Component; import java.util.Collection; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.TimeUnit; /** * redis 工具类 * @author LiWT * @Date 2021-06-22 **/ @SuppressWarnings(value = {"unchecked", "rawtypes"}) @Component public class RedisCache { public RedisTemplate redisTemplate; @Autowired public RedisCache(RedisTemplate redisTemplate){ this.redisTemplate=redisTemplate; this.redisTemplate.setKeySerializer(new StringRedisSerializer()); this.redisTemplate.setValueSerializer(new FastJsonRedisSerializer(Object.class)); } /** * 缓存基本的对象,Integer、String、实体类等 * * @param key 缓存的键值 * @param value 缓存的值 */ public void setCacheObject(final String key, final T value) { redisTemplate.opsForValue().set(key, value); } /** * 缓存基本的对象,Integer、String、实体类等 * * @param key 缓存的键值 * @param value 缓存的值 * @param timeout 时间 * @param timeUnit 时间颗粒度 */ public void setCacheObject(final String key, final T value, final Integer timeout, final TimeUnit timeUnit) { redisTemplate.opsForValue().set(key, value, timeout, timeUnit); } /** * 设置有效时间 * * @param key Redis键 * @param timeout 超时时间 * @return true=设置成功;false=设置失败 */ public boolean expire(final String key, final long timeout) { return expire(key, timeout, TimeUnit.SECONDS); } /** * 设置有效时间 * * @param key Redis键 * @param timeout 超时时间 * @param unit 时间单位 * @return true=设置成功;false=设置失败 */ public boolean expire(final String key, final long timeout, final TimeUnit unit) { return redisTemplate.expire(key, timeout, unit); } /** * 获得缓存的基本对象。 * * @param key 缓存键值 * @return 缓存键值对应的数据 */ public T getCacheObject(final String key) { ValueOperations operation = redisTemplate.opsForValue(); return operation.get(key); } /** * 删除单个对象 * * @param key */ public boolean deleteObject(final String key) { return redisTemplate.delete(key); } /** * 删除集合对象 * * @param collection 多个对象 * @return */ public long deleteObject(final Collection collection) { return redisTemplate.delete(collection); } /** * 缓存List数据 * * @param key 缓存的键值 * @param %values 待缓存的List数据 * @return 缓存的对象 */ public long setCacheList(final String key, final List dataList) { Long count = redisTemplate.opsForList().rightPushAll(key, dataList); return count == null ? 0 : count; } /** * 获得缓存的list对象 * * @param key 缓存的键值 * @return 缓存键值对应的数据 */ public List getCacheList(final String key) { return redisTemplate.opsForList().range(key, 0, -1); } /** * 缓存Set * * @param key 缓存键值 * @param dataSet 缓存的数据 * @return 缓存数据的对象 */ public long setCacheSet(final String key, final Set dataSet) { Long count = redisTemplate.opsForSet().add(key, dataSet); return count == null ? 0 : count; } /** * 获得缓存的set * * @param key * @return */ public Set getCacheSet(final String key) { return redisTemplate.opsForSet().members(key); } /** * 缓存Map * * @param key * @param dataMap */ public void setCacheMap(final String key, final Map dataMap) { if (dataMap != null) { redisTemplate.opsForHash().putAll(key, dataMap); } } /** * 获得缓存的Map * * @param key * @return */ public Map getCacheMap(final String key) { return redisTemplate.opsForHash().entries(key); } /** * 往Hash中存入数据 * * @param key Redis键 * @param hKey Hash键 * @param value 值 */ public void setCacheMapValue(final String key, final String hKey, final T value) { redisTemplate.opsForHash().put(key, hKey, value); } /** * 获取Hash中的数据 * * @param key Redis键 * @param hKey Hash键 * @return Hash中的对象 */ public T getCacheMapValue(final String key, final String hKey) { HashOperations opsForHash = redisTemplate.opsForHash(); return opsForHash.get(key, hKey); } /** * 获取多个Hash中的数据 * * @param key Redis键 * @param hKeys Hash键集合 * @return Hash对象集合 */ public List getMultiCacheMapValue(final String key, final Collection hKeys) { return redisTemplate.opsForHash().multiGet(key, hKeys); } /** * 获得缓存的基本对象列表 * * @param pattern 字符串前缀 * @return 对象列表 */ public Collection keys(final String pattern) { return redisTemplate.keys(pattern); } /** * @param key 增长的key * @return */ public Long increment(final String key) { return redisTemplate.opsForValue().increment(key); } } redis准备工作完成,我们开始实现获取图片验证码,话不多说直接上代码。 package com.blog.controller; import com.blog.utils.RedisCache; import com.blog.utils.VerifyCodeUtils; import lombok.Data; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RestController; import javax.servlet.http.HttpServletResponse; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.util.Base64; import java.util.HashMap; import java.util.Map; import java.util.UUID; import java.util.concurrent.TimeUnit; /** * 验证码操作处理 * @author LiWT * @Date 2021-06-22 */ @RestController public class CaptchaController { /** * 验证码有效期(分钟) */ final Integer CAPTCHA_EXPIRATION = 2; /** * 验证码 redis key */ final String CAPTCHA_CODE_KEY = "captcha_codes:"; @Autowired private RedisCache redisCache; /** * 生成验证码 */ @GetMapping("/captchaImage") public Map getCode(HttpServletResponse response) throws IOException { // 生成随机字串 String verifyCode = VerifyCodeUtils.generateVerifyCode(4); // 唯一标识 String uuid = UUID.randomUUID().toString(); String verifyKey = CAPTCHA_CODE_KEY + uuid; //验证码存入redis redisCache.setCacheObject(verifyKey, verifyCode, CAPTCHA_EXPIRATION, TimeUnit.MINUTES); ByteArrayOutputStream stream = new ByteArrayOutputStream(); //生成图片 int w = 111, h = 36; VerifyCodeUtils.outputImage(w, h, stream, verifyCode); Map result = new HashMap(); try { //jdk1.8提供了新的Base64类 可以直接调用 Base64.Encoder encoder = Base64.getEncoder(); result.put("code", "200"); result.put("uuid", uuid); result.put("verifyCode", verifyCode); result.put("img", encoder.encodeToString(stream.toByteArray())); result.put("msg", "获取验证码成功"); return result; } catch (Exception e) { result.put("code", "500"); result.put("meg", "获取验证码失败,请联系管理人员!"); return result; } finally { stream.close(); } } @Data static class ImageCodeParams { /** * 验证码 */ private String code; /** * 唯一标识 */ private String uuid; } /** * 图片验证码校验示例(该方法仅供验证码校验参考) * * 校验通过直接处理业务逻辑(true) * * 不通过则返回图片验证码错误,让用户重试(false) * @param params 用户传入的验证码及获取图片验证码时的uuid * @return 验证码校验是否通过 true:通过,false:不通过 */ public boolean check(ImageCodeParams params) { String code = redisCache.getCacheObject(CAPTCHA_CODE_KEY + params.getUuid()); if (code == null || "".equals(code) || !code.equalsIgnoreCase(params.getCode())) { redisCache.deleteObject(CAPTCHA_CODE_KEY + params.getUuid()); return false; } return true; } } 这里还依赖一个生成图片验证码的工具类,代码如下,直接ctrl+c、ctrl+v粘进项目中就可以了,记得改一下导入的包噢。 package com.blog.utils; import javax.imageio.ImageIO; import java.awt.*; import java.awt.geom.AffineTransform; import java.awt.image.BufferedImage; import java.io.IOException; import java.io.OutputStream; import java.security.SecureRandom; import java.util.Arrays; import java.util.Random; /** * 验证码工具类 * @author LiWT * @Date 2021-06-22 */ public class VerifyCodeUtils { /** * 使用到Algerian字体,系统里没有的话需要安装字体,字体只显示大写,去掉了1,0,i,o几个容易混淆的字符 */ public static final String VERIFY_CODES = "123456789ABCDEFGHJKLMNPQRSTUVWXYZ"; private static final Random random = new SecureRandom(); /** * 使用系统默认字符源生成验证码 * * @param verifySize 验证码长度 * @return */ public static String generateVerifyCode(int verifySize) { return generateVerifyCode(verifySize, VERIFY_CODES); } /** * 使用指定源生成验证码 * * @param verifySize 验证码长度 * @param sources 验证码字符源 * @return */ public static String generateVerifyCode(int verifySize, String sources) { if (sources == null || sources.length() == 0) { sources = VERIFY_CODES; } int codesLen = sources.length(); Random rand = new Random(System.currentTimeMillis()); StringBuilder verifyCode = new StringBuilder(verifySize); for (int i = 0; i int verifySize = code.length(); BufferedImage image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB); Random rand = new Random(); Graphics2D g2 = image.createGraphics(); g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); Color[] colors = new Color[5]; Color[] colorSpaces = new Color[]{Color.WHITE, Color.CYAN, Color.GRAY, Color.LIGHT_GRAY, Color.MAGENTA, Color.ORANGE, Color.PINK, Color.YELLOW}; float[] fractions = new float[colors.length]; for (int i = 0; i int x = random.nextInt(w - 1); int y = random.nextInt(h - 1); int xl = random.nextInt(6) + 1; int yl = random.nextInt(12) + 1; g2.drawLine(x, y, x + xl + 40, y + yl + 20); } // 添加噪点 float yawpRate = 0.05f;// 噪声率 int area = (int) (yawpRate * w * h); for (int i = 0; i AffineTransform affine = new AffineTransform(); affine.setToRotation(Math.PI / 4 * rand.nextDouble() * (rand.nextBoolean() ? 1 : -1), (w / verifySize) * i + fontSize / 2, h / 2); g2.setTransform(affine); g2.drawChars(chars, i, 1, ((w - 10) / verifySize) * i + 5, h / 2 + fontSize / 2 - 10); } g2.dispose(); ImageIO.write(image, "jpg", os); } private static Color getRandColor(int fc, int bc) { if (fc > 255) { fc = 255; } if (bc > 255) { bc = 255; } int r = fc + random.nextInt(bc - fc); int g = fc + random.nextInt(bc - fc); int b = fc + random.nextInt(bc - fc); return new Color(r, g, b); } private static int getRandomIntColor() { int[] rgb = getRandomRgb(); int color = 0; for (int c : rgb) { color = color rgb[i] = random.nextInt(255); } return rgb; } private static void shear(Graphics g, int w1, int h1, Color color) { shearX(g, w1, h1, color); shearY(g, w1, h1, color); } private static void shearX(Graphics g, int w1, int h1, Color color) { int period = random.nextInt(2); boolean borderGap = true; int frames = 1; int phase = random.nextInt(2); for (int i = 0; i g.setColor(color); g.drawLine((int) d, i, 0, i); g.drawLine((int) d + w1, i, w1, i); } } } private static void shearY(Graphics g, int w1, int h1, Color color) { int period = random.nextInt(40) + 10; // 50; boolean borderGap = true; int frames = 20; int phase = 7; for (int i = 0; i g.setColor(color); g.drawLine(i, (int) d, i, 0); g.drawLine(i, (int) d + h1, i, h1); } } } } CaptchaController中的/captchaImage接口就实现了获取图片验证码,以流的方式返回给前端,前端记得转一下格式就ok了。CaptchaController中的check方法做了验证码的检验,在进行业务逻辑前先进行验证码的校验就可以啦。 到此为止我们的图片验证码实现就完成了,图片验证码使用场景还是蛮多的,该文章记录一下,下次可以直接拿来用了。


【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

      专题文章
        CopyRight 2018-2019 实验室设备网 版权所有