本文章主要探讨React中,Blob格式的图片不展示问题
前端获取后端传递的二进制图片流并转为Blob
const getCaptcha = async () => {
try {
const res = await getCodeUsingPOST();
// 设置倒计时
setDate(Date.now() + 60 * 1000);
const blob = new Blob([res]);
let url = window.URL.createObjectURL(blob);
let captchaImg = document.getElementById('checkImg');
if (captchaImg) {
//@ts-ignore
captchaImg.src = url
captchaImg.onload = function () {
URL.revokeObjectURL(url)
}
}
} catch (e) {
console.log(e)
}
}
Blob不展示原因
1、如果出现请求报错的话,那么就是对响应response进行拦截了
2、如果在网络查看响应没数据的话,那么就是没有配置响应类型
解决方案
问题一:1、如果出现请求报错的话,那么就是对响应response进行拦截了
在request配置文件中配置,配置如果响应路径包含中需要请求的路径,那么返回响应
// 响应拦截器
responseInterceptors: [
(response) => {
// 拦截响应数据,进行个性化处理
const { data } = response as unknown as ResponseStructure;
const path = response.request.responseURL;
// 如果路径中包含以下路径,不拦截,返回响应操作
if(path.includes('/verify/getCode') || path.includes('/verify/checkCode')) {
return response;
} // 这个if判断就是解决问题的代码
if (data.code === 500)
if (data?.code != 0) {
throw new Error(data.message);
}
return response;
},
],
问题二:2、如果在网络查看响应没数据的话,那么就是没有配置响应类型
在定义的请求类型中,配置上responseType:"blob"
/** getCode POST /api/verify/getCode */
export async function getCodeUsingPOST(options?: { [key: string]: any }) {
return request<any>('/api/verify/getCode', {
method: 'POST',
...(options || {}),
responseType: 'blob', // 这个就是解决问题的代码
});
}
另外附上生成验证码的工具类和访问后端获取验证码和检验验证码的Controller
生成验证码工具类 CheckUtils.java
package cn.xiaoxu.project.utils;
import java.awt.*;
import java.awt.geom.AffineTransform;
import java.awt.image.BufferedImage;
import java.util.Random;
/**
* 图形验证码生成
*/
public class CheckUtils {
// 默认验证码字符集
private static final char[] chars = {
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'};
// 默认字符数量
private final Integer SIZE;
// 默认干扰线数量
private final int LINES;
// 默认宽度
private final int WIDTH;
// 默认高度
private final int HEIGHT;
// 默认字体大小
private final int FONT_SIZE;
// 默认字体倾斜
private final boolean TILT;
private final Color BACKGROUND_COLOR;
/**
* 初始化基础参数
*
* @param builder
*/
private CheckUtils(Builder builder) {
SIZE = builder.size;
LINES = builder.lines;
WIDTH = builder.width;
HEIGHT = builder.height;
FONT_SIZE = builder.fontSize;
TILT = builder.tilt;
BACKGROUND_COLOR = builder.backgroundColor;
}
/**
* 实例化构造器对象
*
* @return
*/
public static Builder newBuilder() {
return new Builder();
}
/**
* @return 生成随机验证码及图片
* Object[0]:验证码字符串;
* Object[1]:验证码图片。
*/
public Object[] createImage() {
StringBuffer sb = new StringBuffer();
// 创建空白图片
BufferedImage image = new BufferedImage(WIDTH, HEIGHT, BufferedImage.TYPE_INT_RGB);
// 获取图片画笔
Graphics2D graphic = image.createGraphics();
// 设置抗锯齿
graphic.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
// 设置画笔颜色
graphic.setColor(BACKGROUND_COLOR);
// 绘制矩形背景
graphic.fillRect(0, 0, WIDTH, HEIGHT);
// 画随机字符
Random ran = new Random();
//graphic.setBackground(Color.WHITE);
// 计算每个字符占的宽度,这里预留一个字符的位置用于左右边距
int codeWidth = WIDTH / (SIZE + 1);
// 字符所处的y轴的坐标
int y = HEIGHT * 3 / 4;
for (int i = 0; i < SIZE; i++) {
// 设置随机颜色
graphic.setColor(getRandomColor());
// 初始化字体
Font font = new Font(null, Font.BOLD + Font.ITALIC, FONT_SIZE);
if (TILT) {
// 随机一个倾斜的角度 -45到45度之间
int theta = ran.nextInt(45);
// 随机一个倾斜方向 左或者右
theta = (ran.nextBoolean() == true) ? theta : -theta;
AffineTransform affineTransform = new AffineTransform();
affineTransform.rotate(Math.toRadians(theta), 0, 0);
font = font.deriveFont(affineTransform);
}
// 设置字体大小
graphic.setFont(font);
// 计算当前字符绘制的X轴坐标
int x = (i * codeWidth) + (codeWidth / 2);
// 取随机字符索引
int n = ran.nextInt(chars.length);
// 得到字符文本
String code = String.valueOf(chars[n]);
// 画字符
graphic.drawString(code, x, y);
// 记录字符
sb.append(code);
}
// 画干扰线
for (int i = 0; i < LINES; i++) {
// 设置随机颜色
graphic.setColor(getRandomColor());
// 随机画线
graphic.drawLine(ran.nextInt(WIDTH), ran.nextInt(HEIGHT), ran.nextInt(WIDTH), ran.nextInt(HEIGHT));
}
// 返回验证码和图片
return new Object[]{sb.toString(), image};
}
/**
* 随机取色
*/
private Color getRandomColor() {
Random ran = new Random();
Color color = new Color(ran.nextInt(256), ran.nextInt(256), ran.nextInt(256));
return color;
}
/**
* 构造器对象
*/
public static class Builder {
// 默认字符数量
private int size = 4;
// 默认干扰线数量
private int lines = 10;
// 默认宽度
private int width = 80;
// 默认高度
private int height = 35;
// 默认字体大小
private int fontSize = 25;
// 默认字体倾斜
private boolean tilt = true;
//背景颜色
private Color backgroundColor = Color.LIGHT_GRAY;
public Builder setSize(int size) {
this.size = size;
return this;
}
public Builder setLines(int lines) {
this.lines = lines;
return this;
}
public Builder setWidth(int width) {
this.width = width;
return this;
}
public Builder setHeight(int height) {
this.height = height;
return this;
}
public Builder setFontSize(int fontSize) {
this.fontSize = fontSize;
return this;
}
public Builder setTilt(boolean tilt) {
this.tilt = tilt;
return this;
}
public Builder setBackgroundColor(Color backgroundColor) {
this.backgroundColor = backgroundColor;
return this;
}
public CheckUtils build() {
return new CheckUtils(this);
}
}
}
后端生成验证码Controller
package cn.xiaoxu.project.controller;
import cn.xiaoxu.project.common.BaseResponse;
import cn.xiaoxu.project.common.ErrorCode;
import cn.xiaoxu.project.common.ResultUtils;
import cn.xiaoxu.project.exception.BusinessException;
import cn.xiaoxu.project.utils.CheckUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import org.springframework.web.bind.annotation.*;
import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.OutputStream;
import java.util.concurrent.TimeUnit;
//测试Controller
@RestController
@RequestMapping("verify")
public class VerifyController {
@Autowired
RedisTemplate redisTemplate;
/**
* 生成验证码的接口
*
* @param response Response对象
* @param request Request对象
* @throws Exception
*/
@PostMapping("/getCode")
public void getCode(HttpServletResponse response, HttpServletRequest request) throws Exception {
// 获取到session
HttpSession session = request.getSession();
// 取到sessionid
String id = session.getId();
// 利用图片工具生成图片
// 返回的数组第一个参数是生成的验证码,第二个参数是生成的图片
Object[] objs = CheckUtils.newBuilder()
.setWidth(120) //设置图片的宽度
.setHeight(35) //设置图片的高度
.setSize(6) //设置字符的个数
.setLines(3) //设置干扰线的条数
.setFontSize(25) //设置字体的大小
.setTilt(true) //设置是否需要倾斜
.setBackgroundColor(Color.LIGHT_GRAY) //设置验证码的背景颜色
.build() //构建VerifyUtil项目
.createImage(); //生成图片
// 将验证码存入Session
session.setAttribute("SESSION_VERIFY_CODE_" + id, objs[0]);
// 打印验证码
System.out.println(objs[0]);
// 设置redis值的序列化方式
redisTemplate.setValueSerializer(new StringRedisSerializer());
// 在redis中保存一个验证码最多尝试次数
redisTemplate.opsForValue().set(("VERIFY_CODE_" + id), "3", 1 * 60, TimeUnit.SECONDS);
// 将图片输出给浏览器
BufferedImage image = (BufferedImage) objs[1];
response.setContentType("image/png");
OutputStream os = response.getOutputStream();
try {
ImageIO.write(image, "png", os);
} catch (Exception e) {
e.printStackTrace();
} finally {
os.close();
}
}
/**
* 业务接口包含了验证码的验证
*
* @param code 前端传入的验证码
* @param request Request对象
* @return
*/
@GetMapping("/checkCode")
public BaseResponse<Boolean> checkCode(String code, HttpServletRequest request) {
HttpSession session = request.getSession();
String id = session.getId();
// 将redis中的尝试次数减一
String verifyCodeKey = "VERIFY_CODE_" + id;
long num = redisTemplate.opsForValue().decrement(verifyCodeKey);
// 如果次数次数小于0 说明验证码已经失效
if (num < 0) {
throw new BusinessException(ErrorCode.OPERATION_ERROR, "验证码失效!");
}
// 将session中的取出对应session id生成的验证码
String serverCode = (String) session.getAttribute("SESSION_VERIFY_CODE_" + id);
// 校验验证码
if (null == serverCode || null == code || !serverCode.toUpperCase().equals(code.toUpperCase())) {
throw new BusinessException(ErrorCode.OPERATION_ERROR, "验证码错误!");
}
// 验证通过之后手动将验证码失效
redisTemplate.delete(verifyCodeKey);
// 这里做具体业务相关
return ResultUtils.success(true);
}
}