首页 > 其他分享 >如何实现大文件上传:秒传、断点续传、分片上传

如何实现大文件上传:秒传、断点续传、分片上传

时间:2023-10-27 13:35:22浏览次数:42  
标签:断点续传 文件 param File 分片 fileUploadDTO 上传

前言

文件上传是一个老生常谈的话题了,在文件相对比较小的情况下,可以直接把文件转化为字节流上传到服务器,但在文件比较大的情况下,用普通的方式进行上传,这可不是一个好的办法,毕竟很少有人会忍受,当文件上传到一半中断后,继续上传却只能重头开始上传,这种让人不爽的体验。那有没有比较好的上传体验呢,答案有的,就是下边要介绍的几种上传方式

详细教程

秒传

1、什么是秒传

通俗的说,你把要上传的东西上传,服务器会先做MD5校验,如果服务器上有一样的东西,它就直接给你个新地址,其实你下载的都是服务器上的同一个文件,想要不秒传,其实只要让MD5改变,就是对文件本身做一下修改(改名字不行),例如一个文本文件,你多加几个字,MD5就变了,就不会秒传了.

2、本文实现的秒传核心逻辑

a、利用redis的set方法存放文件上传状态,其中key为文件上传的md5,value为是否上传完成的标志位,

b、当标志位true为上传已经完成,此时如果有相同文件上传,则进入秒传逻辑。如果标志位为false,则说明还没上传完成,此时需要在调用set的方法,保存块号文件记录的路径,其中key为上传文件md5加一个固定前缀,value为块号文件记录路径

分片上传

1.什么是分片上传

分片上传,就是将所要上传的文件,按照一定的大小,将整个文件分隔成多个数据块(我们称之为Part)来进行分别上传,上传完之后再由服务端对所有上传的文件进行汇总整合成原始的文件。

2.分片上传的场景

1.大文件上传

2.网络环境环境不好,存在需要重传风险的场景

断点续传

1、什么是断点续传

断点续传是在下载或上传时,将下载或上传任务(一个文件或一个压缩包)人为的划分为几个部分,每一个部分采用一个线程进行上传或下载,如果碰到网络故障,可以从已经上传或下载的部分开始继续上传或者下载未完成的部分,而没有必要从头开始上传或者下载。本文的断点续传主要是针对断点上传场景。

2、应用场景

断点续传可以看成是分片上传的一个衍生,因此可以使用分片上传的场景,都可以使用断点续传。

3、实现断点续传的核心逻辑

在分片上传的过程中,如果因为系统崩溃或者网络中断等异常因素导致上传中断,这时候客户端需要记录上传的进度。在之后支持再次上传时,可以继续从上次上传中断的地方进行继续上传。

为了避免客户端在上传之后的进度数据被删除而导致重新开始从头上传的问题,服务端也可以提供相应的接口便于客户端对已经上传的分片数据进行查询,从而使客户端知道已经上传的分片数据,从而从下一个分片数据开始继续上传。

4、实现流程步骤

a、方案一,常规步骤

  •  
  •  
    将需要上传的文件按照一定的分割规则,分割成相同大小的数据块;
     
  •  
  •  
    初始化一个分片上传任务,返回本次分片上传唯一标识;
     
  •  
  •  
    按照一定的策略(串行或并行)发送各个分片数据块;
     
  •  
  •  
    发送完成后,服务端根据判断数据上传是否完整,如果完整,则进行数据块合成得到原始文件。
     

b、方案二、本文实现的步骤

  •  
  •  
    前端(客户端)需要根据固定大小对文件进行分片,请求后端(服务端)时要带上分片序号和大小
     
  •  
  •  
    服务端创建conf文件用来记录分块位置,conf文件长度为总分片数,每上传一个分块即向conf文件中写入一个127,那么没上传的位置就是默认的0,已上传的就是Byte.MAX_VALUE 127(这步是实现断点续传和秒传的核心步骤)
     
  •  
  •  
    服务器按照请求数据中给的分片序号和每片分块大小(分片大小是固定且一样的)算出开始位置,与读取到的文件片段数据,写入文件。
     

5、分片上传/断点上传代码实现

a、前端采用百度提供的webuploader的插件,进行分片。因本文主要介绍服务端代码实现,webuploader如何进行分片,具体实现可以查看如下链接:

http://fex.baidu.com/webuploader/getting-started.html

b、后端用两种方式实现文件写入,一种是用RandomAccessFile,如果对RandomAccessFile不熟悉的朋友,可以查看如下链接:


另一种是使用MappedByteBuffer,对MappedByteBuffer不熟悉的朋友,可以查看如下链接进行了解:

https://www.jianshu.com/p/f90866dcbffc

后端进行写入操作的核心代码

a、RandomAccessFile实现方式

@UploadMode(mode = UploadModeEnum.RANDOM_ACCESS)  @Slf4j  public class RandomAccessUploadStrategy extends SliceUploadTemplate {      @Autowired    private FilePathUtil filePathUtil;      @Value("${upload.chunkSize}")    private long defaultChunkSize;      @Override    public boolean upload(FileUploadRequestDTO param) {      RandomAccessFile accessTmpFile = null;      try {        String uploadDirPath = filePathUtil.getPath(param);        File tmpFile = super.createTmpFile(param);        accessTmpFile = new RandomAccessFile(tmpFile, "rw");        //这个必须与前端设定的值一致        long chunkSize = Objects.isNull(param.getChunkSize()) ? defaultChunkSize * 1024 * 1024            : param.getChunkSize();        long offset = chunkSize * param.getChunk();        //定位到该分片的偏移量        accessTmpFile.seek(offset);        //写入该分片数据        accessTmpFile.write(param.getFile().getBytes());        boolean isOk = super.checkAndSetUploadProgress(param, uploadDirPath);        return isOk;      } catch (IOException e) {        log.error(e.getMessage(), e);      } finally {        FileUtil.close(accessTmpFile);      }     return false;    }    }

b、MappedByteBuffer实现方式

@UploadMode(mode = UploadModeEnum.MAPPED_BYTEBUFFER)  @Slf4j  public class MappedByteBufferUploadStrategy extends SliceUploadTemplate {      @Autowired    private FilePathUtil filePathUtil;      @Value("${upload.chunkSize}")    private long defaultChunkSize;      @Override    public boolean upload(FileUploadRequestDTO param) {        RandomAccessFile tempRaf = null;      FileChannel fileChannel = null;      MappedByteBuffer mappedByteBuffer = null;      try {        String uploadDirPath = filePathUtil.getPath(param);        File tmpFile = super.createTmpFile(param);        tempRaf = new RandomAccessFile(tmpFile, "rw");        fileChannel = tempRaf.getChannel();          long chunkSize = Objects.isNull(param.getChunkSize()) ? defaultChunkSize * 1024 * 1024            : param.getChunkSize();        //写入该分片数据        long offset = chunkSize * param.getChunk();        byte[] fileData = param.getFile().getBytes();        mappedByteBuffer = fileChannel  .map(FileChannel.MapMode.READ_WRITE, offset, fileData.length);        mappedByteBuffer.put(fileData);        boolean isOk = super.checkAndSetUploadProgress(param, uploadDirPath);        return isOk;        } catch (IOException e) {        log.error(e.getMessage(), e);      } finally {          FileUtil.freedMappedByteBuffer(mappedByteBuffer);        FileUtil.close(fileChannel);        FileUtil.close(tempRaf);        }        return false;    }    }

c、文件操作核心模板类代码

@Slf4j  public abstract class SliceUploadTemplate implements SliceUploadStrategy {      public abstract boolean upload(FileUploadRequestDTO param);      protected File createTmpFile(FileUploadRequestDTO param) {        FilePathUtil filePathUtil = SpringContextHolder.getBean(FilePathUtil.class);      param.setPath(FileUtil.withoutHeadAndTailDiagonal(param.getPath()));      String fileName = param.getFile().getOriginalFilename();      String uploadDirPath = filePathUtil.getPath(param);      String tempFileName = fileName + "_tmp";      File tmpDir = new File(uploadDirPath);      File tmpFile = new File(uploadDirPath, tempFileName);      if (!tmpDir.exists()) {        tmpDir.mkdirs();      }      return tmpFile;    }      @Override    public FileUploadDTO sliceUpload(FileUploadRequestDTO param) {        boolean isOk = this.upload(param);      if (isOk) {        File tmpFile = this.createTmpFile(param);        FileUploadDTO fileUploadDTO = this.saveAndFileUploadDTO(param.getFile().getOriginalFilename(), tmpFile);        return fileUploadDTO;      }      String md5 = FileMD5Util.getFileMD5(param.getFile());        Map<Integer, String> map = new HashMap<>();      map.put(param.getChunk(), md5);      return FileUploadDTO.builder().chunkMd5Info(map).build();    }      /**     * 检查并修改文件上传进度     */    public boolean checkAndSetUploadProgress(FileUploadRequestDTO param, String uploadDirPath) {        String fileName = param.getFile().getOriginalFilename();      File confFile = new File(uploadDirPath, fileName + ".conf");      byte isComplete = 0;      RandomAccessFile accessConfFile = null;      try {        accessConfFile = new RandomAccessFile(confFile, "rw");        //把该分段标记为 true 表示完成        System.out.println("set part " + param.getChunk() + " complete");        //创建conf文件文件长度为总分片数,每上传一个分块即向conf文件中写入一个127,那么没上传的位置就是默认0,已上传的就是Byte.MAX_VALUE 127        accessConfFile.setLength(param.getChunks());        accessConfFile.seek(param.getChunk());        accessConfFile.write(Byte.MAX_VALUE);          //completeList 检查是否全部完成,如果数组里是否全部都是127(全部分片都成功上传)        byte[] completeList = FileUtils.readFileToByteArray(confFile);        isComplete = Byte.MAX_VALUE;        for (int i = 0; i < completeList.length && isComplete == Byte.MAX_VALUE; i++) {          //与运算, 如果有部分没有完成则 isComplete 不是 Byte.MAX_VALUE          isComplete = (byte) (isComplete & completeList[i]);          System.out.println("check part " + i + " complete?:" + completeList[i]);        }        } catch (IOException e) {        log.error(e.getMessage(), e);      } finally {        FileUtil.close(accessConfFile);      }   boolean isOk = setUploadProgress2Redis(param, uploadDirPath, fileName, confFile, isComplete);      return isOk;    }      /**     * 把上传进度信息存进redis     */    private boolean setUploadProgress2Redis(FileUploadRequestDTO param, String uploadDirPath,        String fileName, File confFile, byte isComplete) {        RedisUtil redisUtil = SpringContextHolder.getBean(RedisUtil.class);      if (isComplete == Byte.MAX_VALUE) {        redisUtil.hset(FileConstant.FILE_UPLOAD_STATUS, param.getMd5(), "true");        redisUtil.del(FileConstant.FILE_MD5_KEY + param.getMd5());        confFile.delete();        return true;      } else {        if (!redisUtil.hHasKey(FileConstant.FILE_UPLOAD_STATUS, param.getMd5())) {          redisUtil.hset(FileConstant.FILE_UPLOAD_STATUS, param.getMd5(), "false");          redisUtil.set(FileConstant.FILE_MD5_KEY + param.getMd5(),              uploadDirPath + FileConstant.FILE_SEPARATORCHAR + fileName + ".conf");        }          return false;      }    }  /**     * 保存文件操作     */    public FileUploadDTO saveAndFileUploadDTO(String fileName, File tmpFile) {        FileUploadDTO fileUploadDTO = null;        try {          fileUploadDTO = renameFile(tmpFile, fileName);        if (fileUploadDTO.isUploadComplete()) {          System.out              .println("upload complete !!" + fileUploadDTO.isUploadComplete() + " name=" + fileName);          //TODO 保存文件信息到数据库          }        } catch (Exception e) {        log.error(e.getMessage(), e);      } finally {        }      return fileUploadDTO;    }  /**     * 文件重命名     *     * @param toBeRenamed 将要修改名字的文件     * @param toFileNewName 新的名字     */    private FileUploadDTO renameFile(File toBeRenamed, String toFileNewName) {      //检查要重命名的文件是否存在,是否是文件      FileUploadDTO fileUploadDTO = new FileUploadDTO();      if (!toBeRenamed.exists() || toBeRenamed.isDirectory()) {        log.info("File does not exist: {}", toBeRenamed.getName());        fileUploadDTO.setUploadComplete(false);        return fileUploadDTO;      }      String ext = FileUtil.getExtension(toFileNewName);      String p = toBeRenamed.getParent();      String filePath = p + FileConstant.FILE_SEPARATORCHAR + toFileNewName;      File newFile = new File(filePath);      //修改文件名      boolean uploadFlag = toBeRenamed.renameTo(newFile);        fileUploadDTO.setMtime(DateUtil.getCurrentTimeStamp());      fileUploadDTO.setUploadComplete(uploadFlag);      fileUploadDTO.setPath(filePath);      fileUploadDTO.setSize(newFile.length());      fileUploadDTO.setFileExt(ext);      fileUploadDTO.setFileId(toFileNewName);        return fileUploadDTO;    }  }

总结

在实现分片上传的过程,需要前端和后端配合,比如前后端的上传块号的文件大小,前后端必须得要一致,否则上传就会有问题。其次文件相关操作正常都是要搭建一个文件服务器的,比如使用fastdfs、hdfs等。

本示例代码在电脑配置为4核内存8G情况下,上传24G大小的文件,上传时间需要30多分钟,主要时间耗费在前端的md5值计算,后端写入的速度还是比较快。如果项目组觉得自建文件服务器太花费时间,且项目的需求仅仅只是上传下载,那么推荐使用阿里的oss服务器,其介绍可以查看官网:

https://help.aliyun.com/product/31815.html

阿里的oss它本质是一个对象存储服务器,而非文件服务器,因此如果有涉及到大量删除或者修改文件的需求,oss可能就不是一个好的选择。

 

参考文章:http://blog.ncmem.com/wordpress/2023/09/21/如何实现大文件上传:秒传、断点续传、分片上传/


 

 

 

 

 

标签:断点续传,文件,param,File,分片,fileUploadDTO,上传
From: https://blog.51cto.com/u_14023400/8053025

相关文章

  • .net 上传附件错误
     错误net::ERR_CONNECTION_ABORTED 导致这种错误的主要原因是上传的文件太大,服务器不能继续读取请求而过早中断链接Failedtoloadresource:theserverrespondedwithastatusof413()开发环境(IISExpress)1073741824=1GB根目录下创建 web.config 文件,内容如下<......
  • php结合webuploader断点续传的实现
    最近公司项目需要用到断点续传,所以记录一下其中的坑使用到的主要技术webuploaderthinkphp5断点续传的思路:客户端:   1.获取文件md5(MD5是文件唯一标识,用来判断是否存在此文件,并且用作分片的文件夹名)   2.将文件分片   3.验证分片是否上传过,上传过直接跳......
  • 每日一题:吃透大文件上传问题(附可运行的前后端源码)
    大文件上传前言在日常开发中,文件上传是常见的操作之一。文件上传技术使得用户可以方便地将本地文件上传到Web服务器上,这在许多场景下都是必需的,比如网盘上传、头像上传等。但是当我们需要上传比较大的文件的时候,容易碰到以下问题:上传时间比较久中间一旦出错就需要重新上传一般服务......
  • 上传文件导致vs调试终止的问题
    最近碰到上传文件后,后台的vs调试自动终止的问题,postman则不会,百思不得其解。最后找到一篇文章(VS上传图片就终止调试问题_输出被调试程序截断_mike0127的博客-CSDN博客)解决了我的问题。 点击VS【工具】-【选项】-【项目和解决方案】-【Web项目】将【在浏览器窗口关闭时停止调试......
  • php结合web uploader插件实现分片上传文件
    这篇文章主要为大家详细介绍了php结合webuploader插件实现分片上传文件,采用大文件分片并发上传,极大的提高了文件上传效率,感兴趣的小伙伴们可以参考一下 最近研究了下大文件上传的方法,找到了webuploaderjs插件进行大文件上传,大家也可以参考这篇文章进行学习:《WebUploader......
  • gitee 上传提示文件过大的暴力解决方法
    因为经常遇到上传文件过大,每次都是重新拉在复制过去,今天无聊就想彻底解决一下这个问题。 Gitee的免费版本只能上传单个文件小于100M利用红色框的命令行查找出是哪个文件,下面红色文字是我查找的文件,然后执行下面命令行,即可上传成功。gitfilter-branch--force--index-filte......
  • vue处理文件流实现上传下载
    1.文件流转base64axios({method:"post",url:"************",responseType:"blob",//必须将返回数据格式更改为blob格式}).then(res=>{//处理返回的文件流数据转为blob对象letblob=......
  • Dynamics CRM中自定义页面实现附件管理包含下载模板、上传、下载、删除
    前言附件使用的DynamicsCRM平台本身的注释表annotation存储,将附件转换成二进制字节流保存到数据库中,因自带的注释在页面中显示附件不够直观,特做了一个单独的附件管理自定义页面,通过CRM自定义按钮打开对话框的方式展示附件列表页面。同时支持下载附件模板,页面为简单的H5+Bootstr......
  • vue中实现上传 ,下载功能
    上传功能(包括上传图片,上传文件)使用element组件库https://element.eleme.cn/#/zh-CN/component/upload<el-upload class="avatar-uploader":disabled="isUpload"action="":show-file-list="false":before-upload="beforeUplo......
  • MarkDown笔记如何上传cnblog
    简介Dotnet-cnblog工具可以配合typora实现自动上传md文件里图片到博客园的图床,这样就不用自己一张张来上传安装过程1.配置NET环境net环境下载地址:https://dotnet.microsoft.com/zh-cn/download/dotnet/5.0下载后安装NET环境,运行cmd命令:dotnet--info查看是否安装成功2.安......