上代码吧:
第一步,先建个对象:
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import javax.validation.constraints.NotBlank;
@Data
@ApiModel("文件上传")
public class UploadFileByBase64Dto {
@NotBlank(message = "上传信息不能为空")
@ApiModelProperty("编码后的base64信息")
String attachContent;
@ApiModelProperty("文件类型 01图片 02语音 03视频")
String attachType;
}
第二步,写个controller:
@PostMapping(value = "/uploadFileBase64ByParam")
@ApiOperation(value = "base64上传文件", notes = "base64上传文件,RequestParam方式接收", httpMethod = "POST")
@ApiImplicitParams({
@ApiImplicitParam(name = "attachContent", value = "编码后的base64信息", required = true, paramType = "query"),
@ApiImplicitParam(name = "attachType", value = "文件类型 01图片 02语音 03视频", required = true, paramType = "query"),
})
public String uploadFileByParam(){
Map<String, Object> params = this.getParams();
if (params == null || StringUtils.isBlank((String) params.get("attachContent"))){
return this.responseJson(ResultUtil.error("参数缺失"),null,null);
}
String attachContent = params.get("attachContent").toString();
String attachType = StringUtils.isBlank((String) params.get("attachType")) ? null : params.get("attachType").toString();
UploadFileByBase64Dto uploadFileByBase64Dto = new UploadFileByBase64Dto();
uploadFileByBase64Dto.setAttachContent(attachContent);
uploadFileByBase64Dto.setAttachType(attachType);
log.info("文件类型入参:"+attachType);
log.info("base64信息入参:"+attachContent);
return this.responseJson(xxxxxService.uploadFileByBase64(uploadFileByBase64Dto), null, null);
}
第三步,service略了;
第四步,serviceImpl代码:
/**
* 文件上传(base64)
*
* @param uploadFileByBase64Dto
* @return
*/
@Override
public ResultUtil uploadFileByBase64(UploadFileByBase64Dto uploadFileByBase64Dto) {
return saveUploadFile(Base64DecodedMultipartFile.base64ToMultipart(uploadFileByBase64Dto.getAttachContent()));
}
private ResultUtil saveUploadFile(MultipartFile file) {
if (Objects.isNull(file)) {
return ResultUtil.error("未获取到文件!");
}
// 获取图片的文件名
String fileName = file.getOriginalFilename();
// 获取图片的扩展名
String extensionName = fileName.substring(fileName.indexOf("."));
// 新的图片文件名 = 获取UUID+图片扩展名
String newFileName = (UUID.randomUUID().toString().replaceAll("-", "")) + extensionName;
File dest = new File(path, newFileName);
if (!dest.getParentFile().exists()) {
dest.getParentFile().mkdirs();
}
Map<String, Object> result = new HashMap<>();
result.put("url", dest.getPath());
//上传到指定目录
try {
file.transferTo(dest);
//增加逻辑:如果是silk音频格式,需要调用服务器ffmpeg插件转换成mp3
if(".silk".equals(extensionName)){
String shellScript= shellScriptPath + " "+dest.getPath()+" " + silkFormat;
Process ps = Runtime.getRuntime().exec(shellScript);
ps.waitFor();
BufferedReader br = new BufferedReader(new InputStreamReader(ps.getInputStream()));
StringBuffer sb = new StringBuffer();
String line;
while ((line = br.readLine()) != null) {
sb.append(line).append("\n");
}
String shellResult = sb.toString();
logger.debug(shellResult);
br.close();
result.put("url", dest.getPath().replace(".silk",".mp3"));
return ResultUtil.ok(result);
}
} catch (IllegalStateException | IOException | InterruptedException e) {
logger.error("上传文件接口执行文件保存出现异常:", e);
return ResultUtil.error("文件保存异常!");
}
result.put("url", dest.getPath());
return ResultUtil.ok(result);
}
第五步,补充个工具类Base64DecodedMultipartFile:
package com.xxxx.micro.common.util;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Paths;
import org.apache.commons.lang3.StringUtils;
import org.springframework.web.multipart.MultipartFile;
import lombok.extern.slf4j.Slf4j;
import sun.misc.BASE64Decoder;
@Slf4j
public class Base64DecodedMultipartFile implements MultipartFile {
private final byte[] imgContent;
private final String header;
public Base64DecodedMultipartFile(byte[] imgContent, String header) {
this.imgContent = imgContent;
this.header = header.split(";")[0];
}
@Override
public String getName() {
// TODO - implementation depends on your requirements
return System.currentTimeMillis() + Math.random() + "." + header.split("/")[1];
}
@Override
public String getOriginalFilename() {
// TODO - implementation depends on your requirements
return System.currentTimeMillis() + (int)Math.random() * 10000 + "." + header.split("/")[1];
}
@Override
public String getContentType() {
// TODO - implementation depends on your requirements
return header.split(":")[1];
}
@Override
public boolean isEmpty() {
return imgContent == null || imgContent.length == 0;
}
@Override
public long getSize() {
return imgContent.length;
}
@Override
public byte[] getBytes() {
return imgContent;
}
@Override
public InputStream getInputStream() {
return new ByteArrayInputStream(imgContent);
}
@Override
public void transferTo(File dest) throws IOException, IllegalStateException {
FileOutputStream out = null;
try {
out = new FileOutputStream(dest);
out.write(imgContent);
out.close();
} catch (Exception e) {
log.error("transferTos:{}",e);
}finally {
if(null!=out) {
try{
out.flush();
}catch(Exception e){
}
try{
out.close();
}catch(Exception e){
}
}
}
}
public static MultipartFile base64ToMultipart(String base64) {
try {
String[] baseStrs = base64.split(",");
BASE64Decoder decoder = new BASE64Decoder();
byte[] b;
log.info("**baseStrs长度:"+baseStrs.length+ "***base64ToMultipart信息:"+baseStrs);
b = decoder.decodeBuffer(baseStrs[1]);
log.info("byte[]--b信息:"+b);
for (int i = 0; i < b.length; ++i) {
if (b[i] < 0) {
b[i] += 256;
}
}
return new Base64DecodedMultipartFile(b, baseStrs[0]);
} catch (IOException e) {
log.error("transferTo:{}",e);
return null;
}
}
/**
* 文件转base64字符串
* @param filePath
* @return
*/
public static String encryptToBase64(String filePath) {
if (filePath == null) {
return null;
}
try {
byte[] b = Files.readAllBytes(Paths.get(filePath));
return org.apache.commons.codec.binary.Base64.encodeBase64String(b);
} catch (IOException e) {
log.error("encryptToBase64:{}",e);
}
return null;
}
}
第六步,需要说明一点,前端传的base64文件流格式,前面的传参attachContent中的值:data:image/jpg;base64,一定要带着,不然后端不知道转成什么文件格式的,代码也会报错。
attachContent的值如下

补充一点:一个很奇怪的问题,在前端将图片转为base64后,字符串是正常的,可是post请求到后台,发现base64字符串里的"+"号被替换为空格,导致图片保存后,打开失败,无法查看。
问题就是,前端将字符串里的"+"号字符转换一下就好
data.replace(/+/g,"%2B");