资讯详情

base64上传文件,后端接受处理(四)

将文件(图片、视频、音频、压缩包)放在前端base以64流的形式上传文件,转移后端接口,上传。

上代码吧: 第一步是建立一个对象:

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");

标签: 8fu8hu8h8fub传感器ejx115a流量变送器

锐单商城拥有海量元器件数据手册IC替代型号,打造 电子元器件IC百科大全!

锐单商城 - 一站式电子元器件采购平台