文件上传是每个项目都会用到的基础功能,这里讲解下实现过程和一些最佳实践

接收方式

后台如何接收前台传过来的文件呢?其实很简单,用MultipartFile就能接收

@PostMapping("file")
@ApiOperation(value = "上传文件")
public Result uploadFile(@RequestParam("file") MultipartFile file) {
    return resultOk(uploadService.upload(file));
}

保存文件

接收到文件后就需要保存文件到服务器了,直接调用MultipartFile的transferTo方法就能保存,没啥难度,关键是如果指定保存目录。很多人喜欢用当前日期来生成目录保存文件,比如生成一个名为20210810的目录。这种方式有个不好的地方就是文件不均衡,可能有的目录有很多文件,有的目录却没有文件。

一个目录放入太多文件会造成检索缓慢,我们生活中应该有这样的经验,当打开一个文件超多的目录鼠标会显示加载中,过好一会才能看到文件。

那如何才能均衡的存放文件呢?

可以随机生成一个2字节整数,16进制最小为0x0000,最大为0xFFFF,可以把低位作为一级文件名,高位作为二级文件名,比如文件名为ff/ff,这样如果每个文件夹保存256个文件的话,理论可以均衡保存1600万个文件。

如果想均衡保存更多文件咋办?那就取4字节,最大为0xFFFFFFFF,生成4层目录,可均衡保存万亿个文件。当然这么大数量的文件已经不推荐单机保存了,需要考虑HDFS或者OSS。

public String upload(MultipartFile file) {
    try {
        //获取原始文件名
        String originalName = file.getOriginalFilename();
        //获取原始文件后缀
        String extension = FileUtils.getExtention(originalName);
        //生成新的文件名,保留原有后缀
        String fileName = UUIDUtils.getUUID() + "." + extension;
        int pathCode = RandomUtils.nextInt(65535);
        //取低位1字节
        int v = pathCode & 0xFF;
        //取次低位1字节
        int h = pathCode >> 8 & 0xFF;
        String relativeDir = HexUtils.bytesToHexString((byte) v).toLowerCase() + "/" + HexUtils.bytesToHexString((byte) h).toLowerCase() + "/";
        File dir = new File(systemConfig.getUploadPath() + relativeDir);
        if (!dir.exists()) {
            dir.mkdirs();
        }
        File saveFile = new File(dir, fileName);
        file.transferTo(saveFile);
        return systemConfig.getFileBaseUrl() + "file/" + relativeDir + fileName;
    } catch (IOException e) {
        throw new ServiceFailException("上传文件失败", e);
    }
}

保存展示

springboot 文件上传 携带参数 springboot接收文件上传_多文件

 

springboot 文件上传 携带参数 springboot接收文件上传_保存文件_02

 支持URL访问

这个时候虽然保存成功了,但是无法通过URL访问,因为没建立资源的映射关系,下面的uploadPath是本地文件保存路径

@Configuration
public class InterceptorConfig implements WebMvcConfigurer {
    @Autowired
    private SystemConfig systemConfig;

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        //定义静态资源路径
        registry.addResourceHandler("/file/**").addResourceLocations("file:" + systemConfig.getUploadPath());
    }
}

url演示

springboot 文件上传 携带参数 springboot接收文件上传_文件名_03

 文件上传大小限制

SpringBoot默认上传文件大小限制为1MB,这个肯定不够的,需要在配置文件里改下

servlet:
    multipart:
      max-file-size: 100MB
      max-request-size: 100MB