写一下后端spring项目经常要做的功能,实现图片上传和下载,这里也把前端代码附上了。可能算是个简单版的,我这里图片上传都存在当前项目的根目录resource下了。
这里包含了,上传文件、下载文件(下载文件流、获取base64),service中还有个文件流转base64的工具方法。
下面是后端代码
Controller
package com.wft.controller;import com.wft.model.ActionResult;
import com.wft.service.TestService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.io.InputStreamResource;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;import java.io.*;@RestController
@RequestMapping("/file")
public class TestController {@AutowiredTestService testService;/*** 文件上传* @param file* @return* @throws IOException*/@PostMapping("/upload")public ActionResult uploadTest(@RequestParam("file") MultipartFile file) throws IOException {return testService.upload(file);}/*** 文件下载* @param name* @return* @throws FileNotFoundException*/@GetMapping("/download/{name}")@CrossOriginpublic ResponseEntity<InputStreamResource> downloadTest(@PathVariable("name") String name) throws FileNotFoundException {return testService.download(name);}/*** 获取文件的base64编码* @param name* @return* @throws IOException*/@GetMapping("/getBase64/{name}")public ActionResult getBase64(@PathVariable("name") String name) throws IOException {return testService.getBase64(name);}}
Service接口就不贴了哈
ServiceImpl
package com.wft.service.impl;import com.wft.model.ActionResult;
import com.wft.service.TestService;
import org.springframework.core.io.InputStreamResource;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Service;
import org.springframework.util.ResourceUtils;
import org.springframework.web.multipart.MultipartFile;import java.io.*;
import java.nio.file.Files;
import java.util.Base64;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;@Service
public class TestServiceImpl implements TestService {/*** 文件上传逻辑* @param file* @return* @throws IOException*/@Overridepublic ActionResult upload(MultipartFile file) throws IOException {if(file.isEmpty()) {return ActionResult.fail("文件不能为空");}// 上传的文件名称String originalFilename = file.getOriginalFilename();// 文件后缀String suffix = originalFilename.substring(originalFilename.lastIndexOf("."));// 使用uuid当前文件名存储,防止名称相同覆盖String uuid = UUID.randomUUID().toString();String fileName = uuid + suffix;// 文件保存的路径(我这里存在当前项目下,所以获取当前项目的绝对路径,然后拼接上图片存放的文件夹的名称)String savePath = ResourceUtils.getURL("resource").getPath();// 判断是否存在resource目录, 没有则创建File dir = new File(savePath);if(dir != null && !dir.exists()) {dir.mkdir();}savePath = savePath + File.separator + fileName;// 文件上传file.transferTo(new File(savePath));// 将文件存储的名称uuid和原文件名称返回给前端Map<String, Object> map = new HashMap<>();map.put("id", uuid);map.put("name", originalFilename);// 将下载图片的接口(返回文件流)路径返回给前端,前端直接将服务器地址拼上该链接即可回显图片map.put("url", "/file/download/" + fileName);return ActionResult.success("上传成功", map);}/*** 文件下载逻辑(文件流)* @param name* @return* @throws FileNotFoundException*/@Overridepublic ResponseEntity<InputStreamResource> download(String name) throws FileNotFoundException {String path = ResourceUtils.getURL("resource").getPath() + File.separator + name;File file = new File(path);if(!file.exists()) {return new ResponseEntity<>(HttpStatus.NOT_FOUND);}// 创建输入流InputStream inputStream = new FileInputStream(file);// 设置HTTP头部信息HttpHeaders headers = new HttpHeaders();System.out.println(file.getName() + "---->>>文件名");headers.add(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=" + file.getName());// 返回文件流ResponseEntity<InputStreamResource> body = ResponseEntity.ok().headers(headers).contentType(MediaType.APPLICATION_OCTET_STREAM).body(new InputStreamResource(inputStream));return body;}/*** 获取文件的base64编码* @param name* @return* @throws IOException*/@Overridepublic ActionResult getBase64(String name) throws IOException {String path = ResourceUtils.getURL("resource").getPath() + File.separator + name;File file = new File(path);if(!file.exists()) {return ActionResult.fail("文件不存在");}byte[] fileContent = Files.readAllBytes(file.toPath());String base64 = Base64.getEncoder().encodeToString(fileContent);base64 = "data:image/png;base64," + base64;return ActionResult.success((Object) base64);}/*** 将文件流转为base64编码(工具方法, service中接口没有该方法)* @param response* @return* @throws IOException*/public String streamToBase64(ResponseEntity<InputStreamResource> response) throws IOException {// 获取InputStreamResource对象InputStreamResource resource = response.getBody();if (resource == null) {throw new IllegalArgumentException("Response body is null");}try (InputStream inputStream = resource.getInputStream();ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream()) {byte[] buffer = new byte[1024];int bytesRead;while ((bytesRead = inputStream.read(buffer)) != -1) {byteArrayOutputStream.write(buffer, 0, bytesRead);}// 将InputStream转换为字节数组byte[] fileBytes = byteArrayOutputStream.toByteArray();// 使用Base64编码器进行编码String base64Encoded = Base64.getEncoder().encodeToString(fileBytes);base64Encoded = "data:image/png;base64," + base64Encoded;return base64Encoded;}}
}
代码里面也都写注释了,大家一看应该就明白了。
简单说一下思路把,存图片的时候,我是以uuid当作图片的名称存储的,这样即便是前端是两个文件,但是名称一样,上传之后也不会覆盖掉原来的图片。
然后图片上传完之后,我把回显图片的路径返回给前端了,前端可以使用服务器地址(当然开发环境下会有跨域问题,一般会直接用前缀)拼上返回的这个路径即可回显图片。
返回的这个url其实就是后端编写好的一个接口,返回的是个文件流,前端直接将完整的请求的后端的路径放在img标签的src上,其实就相当于发送了请求,所以这里注意这种方式回显要求后端将改接口放在白名单中(即该接口不需要token校验),否则前端就不能像正常路径一样直接放在img的src上回显,就要像普通的接口一样调用接口,然后通过URL.createObjectURL(new Blob(res))的方式转为路径再复制给src。
然后我上面封装了了个ActionResult的返回给前端的包装类,也在这贴一下吧:
ActionResult
package com.wft.model;import lombok.Data;@Data
public class ActionResult<T> {private Integer code;private String msg;private T data;public static ActionResult success() {ActionResult jsonData = new ActionResult();jsonData.setCode(200);jsonData.setMsg("success");return jsonData;}public static ActionResult success(String msg) {ActionResult jsonData = new ActionResult();jsonData.setCode(200);jsonData.setMsg(msg);return jsonData;}public static ActionResult success(Object object) {ActionResult jsonData = new ActionResult();jsonData.setData(object);jsonData.setCode(200);jsonData.setMsg("success");return jsonData;}public static ActionResult success(String msg, Object object) {ActionResult jsonData = new ActionResult();jsonData.setData(object);jsonData.setCode(200);jsonData.setMsg(msg);return jsonData;}public static ActionResult fail(Integer code, String message) {ActionResult jsonData = new ActionResult();jsonData.setCode(code);jsonData.setMsg(message);return jsonData;}public static ActionResult fail(String msg, String data) {ActionResult jsonData = new ActionResult();jsonData.setMsg(msg);jsonData.setData(data);return jsonData;}public static ActionResult fail(String msg) {ActionResult jsonData = new ActionResult();jsonData.setMsg(msg);jsonData.setCode(400);return jsonData;}
}
接下来再贴一下前端代码:
<template><div class="wft-test"><el-uploadclass="avatar-uploader":action="baseURL + '/file/upload'":show-file-list="false":on-success="handleAvatarSuccess"><img v-if="imageUrl" :src="baseURL + imageUrl" class="avatar" /><el-icon v-else class="avatar-uploader-icon"><Plus /></el-icon></el-upload><!-- 测试下载图片 获取文件流 --><el-button @click="testDownload('12b83c8d-3d54-420d-a191-bd750fa571c4.png')">测试下载图片流</el-button><!-- 测试下载图片 获取base64 --><el-button @click="testDownloadBase64('12b83c8d-3d54-420d-a191-bd750fa571c4.png')">测试下载图片base64</el-button><img style="width: 200px;height: 200px;" v-if="imgBase64" :src="imgBase64" alt=""></div>
</template>
<script setup lang='ts'>
import { ref } from 'vue';
import request from '@/utils/request';const imageUrl = ref("");
const imgBase64 = ref("");// 上传成功回调
function handleAvatarSuccess(res: any) {if(res.code == 200) {imageUrl.value = res.data.url // 正常成功回显}
}/*** 测试下载图片(获取文件流)* @param fileName 文件名*/
function testDownload(fileName: string) {request({url: `/file/download/${fileName}`,method: 'get',responseType: 'blob'}).then((res: any) => {const link = document.createElement('a')link.href = URL.createObjectURL(new Blob([res]))link.download = 'test.png'document.body.appendChild(link)link.click()document.body.removeChild(link)})
}/*** 测试下载图片(获取文件base64编码)* @param fileName */
function testDownloadBase64(fileName: string) {request({url: `/file/getBase64/${fileName}`,method: 'get'}).then((res: any) => {if(res.code == 200) {imgBase64.value = res.data}})
}</script>
<style scoped>
.wft-test {width: 100%;height: 100%;
}.avatar-uploader .avatar {width: 178px;height: 178px;display: block;
}
</style><style>
.avatar-uploader .el-upload {border: 1px dashed var(--el-border-color);border-radius: 6px;cursor: pointer;position: relative;overflow: hidden;transition: var(--el-transition-duration-fast);
}.avatar-uploader .el-upload:hover {border-color: var(--el-color-primary);
}.el-icon.avatar-uploader-icon {font-size: 28px;color: #8c939d;width: 178px;height: 178px;text-align: center;
}
</style>