一、问题的原由
以往的项目中要访问图片直接写个url去指向项目中的存储图片的文件夹,然后写访问哪个图片。比如:
http://localhost:8080/ProductSys/img/1.png
但是,在我们的ssm项目中,一般都会使用拦截器(HandlerInterceptor接口的实现类),在用户做任何操作时先拦截看看用户是否登录,用户的权限是否合格等等。
我们会设置所有请求的请求头内容为一个token,请求一般都是在请求体中设置参数,但我们登录验证一般放在头部,拦截器就拦截使用token然后就是业务判断了…
至于如何给所有请求的请求头都添加一个内容,就可以使用axios添加一个拦截器,添加一个请求拦截器,用axios请求时就可以对所有的请求前设置内容了。
可是 http://localhost:8080/ProductSys/img/1.png 它是一个请求,但它不是使用axios的请求,我们只对所有使用了axios的请求设置了请求头内容,这导致了这个图片请求时请求头没有token,而拦截器对它拦截时就获取不到这个token,会直接放行。让我们访问不到。
二、解决方法
2.1 判断请求路径
在拦截器中通过HttpServletRequest对象获取请求路径,判断请求路径是否包含img等资源文件夹名称,包含就放行。不包含就没有请求资源文件夹下的资源,继续我们的获取token操作。 如请求路径是:
http://localhost:8080/ProductSys/img/1.png
我们拦截器类中可以这样写
String requestURI = request.getRequestURI().toString();
if(requestURI.contains("img") || requestURI.contains("toLogin") || requestURI.contains("login")){
return true;
}
这样有一个不坏的地方就是不管用户登没登录,权限够不够,它都可以通过请求路径直接访问我们的资源文件夹下的图片的资源。
2.2 对图片进行编码
- 对图片进行二进制流字符编译
- 将编译的二进制数据转换为base64编码的数据
- 将base64编码的数据送给前端
这种方式,img标签的src就不是url了,而是
<img src="“data:img/png;base64,”+base64编码数据">
流程:
- 通过axios发送图书请求
- axios就会对这个请求的请求头设置一个token
- 拦截器进行拦截解析token
- 解析token通过
- 进入将图片转换为base64编码的数据字符方法
- 响应base64编码的数据字符给前端
- axios使用.then(res){ xxxx } 接收
这种方式就可以在用户没登录,权限不够的情况下,它就不可以访问我们的资源文件夹下的图片的资源。
Controller(控制器)层
@RequestMapping("/uploadFile")
public String uploadFile(@RequestParam("img") MultipartFile multipartFile,@RequestParam("id") String id, HttpServletRequest request) throws IOException {
String rootPath = request.getSession().getServletContext().getRealPath("/upload");
PictureUtils.uploadFile(multipartFile,id,rootPath);
return "success";
}
@RequestMapping("/downFile/{id}")
public String downFile(@PathVariable("id") String id,HttpServletRequest request) throws Exception{
return PictureUtils.downFile(id,request);
}
PictureUtils类(工具类)
package com.apps.utils;
import org.springframework.web.multipart.MultipartFile;
import sun.misc.BASE64Encoder;
import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletRequest;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
public class PictureUtils {
public static void uploadFile(MultipartFile multipartFile, String id, String rootPath) throws IOException {
String newFileName = id + ".png";
File newFile = new File(rootPath + File.separator + newFileName);
if(!newFile.getParentFile().exists()) {
newFile.getParentFile().mkdirs();
}
multipartFile.transferTo(newFile);
}
public static String downFile(String id, HttpServletRequest request)throws IOException {
BASE64Encoder encoder = new sun.misc.BASE64Encoder();
Map<String, File> fileMap = new HashMap<>();
String[] supportFormat = {"gif","jpg","png"};
for (String format : supportFormat) {
String fileName = request.getSession().getServletContext().getRealPath("upload")+ File.separator + id + "." + format;
File f = new File(fileName);
if(f.exists()) fileMap.put(format,f);
}
File file = new File(request.getSession().getServletContext().getRealPath("upload") + File.separator +"no.png");;
String format = "png";
if(!(fileMap.size() == 0)){
Set<String> strings = fileMap.keySet();
one : for (String string : strings) {
file = fileMap.get(string);
format = string;
break one;
}
}
BufferedImage bi = ImageIO.read(file);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ImageIO.write(bi, format, baos);
byte[] bytes = baos.toByteArray();
return encoder.encodeBuffer(bytes).trim();
}
}
|