Java大文件上传、分片上传、多文件上传、断点续传、上传文件minio、分片上传minio等解决方案

一、上传说明

         文件上传花样百出,根据不同场景使用不同方案进行实现尤为必要。通常开发过程中,文件较小,直接将文件转化为字节流上传到服务器,但是文件较大时,用普通的方法上传,显然效果不是很好,当文件上传一半中断再次上传时,发现需要重新开始,这种体验不是很爽,下面介绍几种好一点儿的上传方式。

        这里讲讲如何在Spring boot 编写上传代码,如有问题可以在下留言,我并在文章末尾附上Java上传源码供大家下载。

分片上传

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

断点续传

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

二、Redis启动安装

Redis安装包分为 Windows 版和 Linux 版:
Windows版下载地址:https://github.com/microsoftarchive/redis/releases
Linux版下载地址: https://download.redis.io/releases/
我当前使用的Windows版本:

三、minio下载启动

windows版本可以参考我之前的文档:window10安装minio_minio windows安装-CSDN博客

启动会提示:

以上是密码设置问题需要修改如下:

set MINIO_ROOT_USER=admin
set MINIO_ROOT_PASSWORD=12345678

启动成功后会输出相应地址

四、上传后端Java代码

        后端采用Spring boot项目结构,主要代码如下:

  /**
     * 单文件上传
     * 直接将传入的文件通过io流形式直接写入(服务器)指定路径下
     *
     * @param file 上传的文件
     * @return
     */
    @Override
    public ResultEntity<Boolean> singleFileUpload(MultipartFile file) {
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";
        File dir = new File(filePath);
        if (!dir.exists()) dir.mkdir();

        if (file == null) {
            return ResultEntity.error(false, "上传文件为空!");
        }
        InputStream fileInputStream = null;
        FileOutputStream fileOutputStream = null;
        try {
            String filename = file.getOriginalFilename();
            fileOutputStream = new FileOutputStream(filePath + filename);
            fileInputStream = file.getInputStream();

            byte[] buf = new byte[1024 * 8];
            int length;
            while ((length = fileInputStream.read(buf)) != -1) {//读取fis文件输入字节流里面的数据
                fileOutputStream.write(buf, 0, length);//通过fos文件输出字节流写出去
            }
            log.info("单文件上传完成!文件路径:{},文件名:{},文件大小:{}", filePath, filename, file.getSize());
            return ResultEntity.success(true, "单文件上传完成!");
        } catch (IOException e) {
            return ResultEntity.error(true, "单文件上传失败!");
        } finally {
            try {
                if (fileOutputStream != null) {
                    fileOutputStream.close();
                    fileOutputStream.flush();
                }
                if (fileInputStream != null) {
                    fileInputStream.close();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    /**
     * 多文件上传
     * 直接将传入的多个文件通过io流形式直接写入(服务器)指定路径下
     * 写入指定路径下是通过多线程进行文件写入的,文件写入线程执行功能就和上面单文件写入是一样的
     *
     * @param files 上传的所有文件
     * @return
     */
    @Override
    public ResultEntity<Boolean> multipleFileUpload(MultipartFile[] files) {
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";
        File dir = new File(filePath);
        if (!dir.exists()) dir.mkdir();

        if (files.length == 0) {
            return ResultEntity.error(false, "上传文件为空!");
        }
        ArrayList<String> uploadFiles = new ArrayList<>();
        try {

            ArrayList<Future<String>> futures = new ArrayList<>();
            //使用多线程来完成对每个文件的写入
            for (MultipartFile file : files) {
                futures.add(partMergeTask.submit(new MultipleFileTaskExecutor(filePath, file)));
            }

            //这里主要用于监听各个文件写入线程是否执行结束
            int count = 0;
            while (count != futures.size()) {
                for (Future<String> future : futures) {
                    if (future.isDone()) {
                        uploadFiles.add(future.get());
                        count++;
                    }
                }
                Thread.sleep(1);
            }
            log.info("多文件上传完成!文件路径:{},文件信息:{}", filePath, uploadFiles);
            return ResultEntity.success(true, "多文件上传完成!");
        } catch (Exception e) {
            log.error("多文件分片上传失败!", e);
            return ResultEntity.error(true, "多文件上传失败!");
        }

    }

    /**
     * 单文件分片上传
     * 直接将传入的文件分片通过io流形式写入(服务器)指定临时路径下
     * 然后判断是否分片都上传完成,如果所有分片都上传完成的话,就把临时路径下的分片文件通过流形式读入合并并从新写入到(服务器)指定文件路径下
     * 最后删除临时文件和临时文件夹,临时文件夹是通过文件的uuid进行命名的
     *
     * @param filePart  分片文件
     * @param partIndex 当前分片值
     * @param partNum   所有分片数
     * @param fileName  当前文件名称
     * @param fileUid   当前文件uuid
     * @return
     */
    @Override
    public ResultEntity<Boolean> singleFilePartUpload(MultipartFile filePart, Integer partIndex, Integer partNum, String fileName, String fileUid) {
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";//文件存放路径
        String tempPath = filePath + "temp\\" + fileUid;//临时文件存放路径
        File dir = new File(tempPath);
        if (!dir.exists()) dir.mkdirs();

        //生成一个临时文件名
        String tempFileNamePath = tempPath + "\\" + fileName + "_" + partIndex + ".part";
        try {
            //将分片存储到临时文件夹中
            filePart.transferTo(new File(tempFileNamePath));

            File tempDir = new File(tempPath);
            File[] tempFiles = tempDir.listFiles();

            one:
            if (partNum.equals(Objects.requireNonNull(tempFiles).length)) {
                //需要校验一下,表示已有异步程序正在合并了;如果是分布式这个校验可以加入redis的分布式锁来完成
                if (isMergePart.get(fileUid) != null) {
                    break one;
                }
                isMergePart.put(fileUid, tempFiles.length);
                System.out.println("所有分片上传完成,预计总分片:" + partNum + "; 实际总分片:" + tempFiles.length);

                FileOutputStream fileOutputStream = new FileOutputStream(filePath + fileName);
                //这里如果分片很多的情况下,可以采用多线程来执行
                for (int i = 0; i < partNum; i++) {
                    //读取分片数据,进行分片合并
                    FileInputStream fileInputStream = new FileInputStream(tempPath + "\\" + fileName + "_" + i + ".part");
                    byte[] buf = new byte[1024 * 8];//8MB
                    int length;
                    while ((length = fileInputStream.read(buf)) != -1) {//读取fis文件输入字节流里面的数据
                        fileOutputStream.write(buf, 0, length);//通过fos文件输出字节流写出去
                    }
                    fileInputStream.close();
                }
                fileOutputStream.flush();
                fileOutputStream.close();

                // 删除临时文件夹里面的分片文件 如果使用流操作且没有关闭输入流,可能导致删除失败
                for (int i = 0; i < partNum; i++) {
                    boolean delete = new File(tempPath + "\\" + fileName + "_" + i + ".part").delete();
                    File file = new File(tempPath + "\\" + fileName + "_" + i + ".part");
                }
                //在删除对应的临时文件夹
                if (Objects.requireNonNull(tempDir.listFiles()).length == 0) {
                    tempDir.delete();
                }
                isMergePart.remove(fileUid);
            }

        } catch (Exception e) {
            log.error("单文件分片上传失败!", e);
            return ResultEntity.error(false, "单文件分片上传失败");
        }
        //通过返回成功的分片值,来验证分片是否有丢失
        return ResultEntity.success(true, partIndex.toString());
    }

    /**
     * 多文件分片上传
     * 先将所有文件分片读入到(服务器)指定临时路径下,每个文件的分片文件的临时文件夹都是已文件的uuid进行命名的
     * 然后判断对已经上传所有分片的文件进行合并,此处是通过多线程对每一个文件的分片文件进行合并的
     * 最后对已经合并完成的分片临时文件和文件夹进行删除
     *
     * @param filePart  分片文件
     * @param partIndex 当前分片值
     * @param partNum   总分片数
     * @param fileName  当前文件名称
     * @param fileUid   当前文件uuid
     * @return
     */
    @Override
    public ResultEntity<String> multipleFilePartUpload(MultipartFile filePart, Integer partIndex, Integer partNum, String fileName, String fileUid) {
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";//文件存放路径
        String tempPath = filePath + "temp\\" + fileUid;//临时文件存放路径
        File dir = new File(tempPath);
        if (!dir.exists()) dir.mkdirs();
        //生成一个临时文件名
        String tempFileNamePath = tempPath + "\\" + fileName + "_" + partIndex + ".part";
        try {
            filePart.transferTo(new File(tempFileNamePath));

            File tempDir = new File(tempPath);
            File[] tempFiles = tempDir.listFiles();
            //如果临时文件夹中分片数量和实际分片数量一致的时候,就需要进行分片合并
            one:
            if (partNum.equals(tempFiles.length)) {
                //需要校验一下,表示已有异步程序正在合并了;如果是分布式这个校验可以加入redis的分布式锁来完成
                if (isMergePart.get(fileUid) != null) {
                    break one;
                }
                isMergePart.put(fileUid, tempFiles.length);
                System.out.println(fileName + ":所有分片上传完成,预计总分片:" + partNum + "; 实际总分片:" + tempFiles.length);

                //使用多线程来完成对每个文件的合并
                Future<Integer> submit = partMergeTask.submit(new PartMergeTaskExecutor(filePath, tempPath, fileName, partNum));
                System.out.println("上传文件名:" + fileName + "; 总大小:" + submit.get());
                isMergePart.remove(fileUid);
            }
        } catch (Exception e) {
            log.error("{}:多文件分片上传失败!", fileName, e);
            return ResultEntity.error("", "多文件分片上传失败");
        }
        //通过返回成功的分片值,来验证分片是否有丢失
        return ResultEntity.success(partIndex.toString(), fileUid);
    }

    /**
     * 多文件(分片)秒传
     * 通过对比已有的文件分片md5值和需要上传文件分片的MD5值,
     * 在文件分片合并的时候,对已有的文件进行地址索引,对没有的文件进行临时文件写入
     * 最后合并的时候根据不同的文件分片进行文件读取写入
     *
     * @param filePart  上传没有的分片文件
     * @param fileInfo  当前分片文件相关信息
     * @param fileOther 已存在文件分片相关信息
     * @return
     */
    @Override
    public ResultEntity<String> multipleFilePartFlashUpload(MultipartFile filePart, String fileInfo, String fileOther) {
        DiskFileIndexVo upFileInfo = JSONObject.parseObject(fileInfo, DiskFileIndexVo.class);
        List<DiskFileIndexVo> notUpFileInfoList = JSON.parseArray(fileOther, DiskFileIndexVo.class);
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";//文件存放路径
        //正常情况下,这个临时文件也应该放入(服务器)非临时文件夹中,这样方便下次其他文件上传查找是否曾经上传过类似的
        //当前demo是单独存放在临时文件夹中,文件合并完成之后直接删除的
        String tempPath = filePath + "temp\\" + upFileInfo.getFileUid();//临时文件存放路径

        File dir = new File(tempPath);
        if (!dir.exists()) dir.mkdirs();
        //生成一个临时文件名
        String tempFileNamePath = tempPath + "\\" + upFileInfo.getFileName() + "_" + upFileInfo.getPartIndex() + ".part";

        try {
            filePart.transferTo(new File(tempFileNamePath));

            File tempDir = new File(tempPath);
            File[] tempFiles = tempDir.listFiles();
            notUpFileInfoList = notUpFileInfoList.stream().filter(e ->
                    upFileInfo.getFileUid().equals(e.getFileUid())).collect(Collectors.toList());
            //如果临时文件夹中分片数量和实际分片数量一致的时候,就需要进行分片合并
            one:
            if ((upFileInfo.getPartNum() - notUpFileInfoList.size()) == tempFiles.length) {
                //需要校验一下,表示已有异步程序正在合并了;如果是分布式这个校验可以加入redis的分布式锁来完成
                if (isMergePart.get(upFileInfo.getFileUid()) != null) {
                    break one;
                }
                isMergePart.put(upFileInfo.getFileUid(), tempFiles.length);
                System.out.println(upFileInfo.getFileName() + ":所有分片上传完成,预计总分片:" + upFileInfo.getPartNum()
                        + "; 实际总分片:" + tempFiles.length + "; 已存在分片数:" + notUpFileInfoList.size());

                //使用多线程来完成对每个文件的合并
                Future<Integer> submit = partMergeTask.submit(
                        new PartMergeFlashTaskExecutor(filePath, upFileInfo, notUpFileInfoList));
                isMergePart.remove(upFileInfo.getFileUid());
            }
        } catch (Exception e) {
            log.error("{}:多文件(分片)秒传失败!", upFileInfo.getFileName(), e);
            return ResultEntity.error("", "多文件(分片)秒传失败!");
        }
        //通过返回成功的分片值,来验证分片是否有丢失
        return ResultEntity.success(upFileInfo.getPartIndex().toString(), upFileInfo.getFileUid());
    }

    /**
     * 根据传入需要上传的文件片段的md5值来对比服务器中的文件的md5值,将已有对应的md5值的文件过滤出来,
     * 通知前端或者自行出来这些文件,即为不需要上传的文件分片,并将已有的文件分片地址索引返回给前端进行出来
     *
     * @param upLoadFileListMd5 原本需要上传文件的索引分片信息
     * @return
     */
    @Override
    public ResultEntity<List<DiskFileIndexVo>> checkDiskFile(List<DiskFileIndexVo> upLoadFileListMd5) {
        List<DiskFileIndexVo> notUploadFile;
        try {
            //后端服务器已经存在的分片md5值集合
            List<DiskFileIndexVo> diskFileMd5IndexList = diskFileIndexVos;

            notUploadFile = upLoadFileListMd5.stream().filter(uf -> diskFileMd5IndexList.stream().anyMatch(
                    df -> {
                        if (df.getFileMd5().equals(uf.getFileMd5())) {
                            uf.setFileIndex(df.getFileName());//不需要上传文件的地址索引
                            return true;
                        }
                        return false;
                    })).collect(Collectors.toList());
            log.info("过滤出不需要上传的文件分片:{}", notUploadFile);
        } catch (Exception e) {
            log.error("上传文件检测异常!", e);
            return ResultEntity.error("上传文件检测异常!");
        }
        return ResultEntity.success(notUploadFile);
    }

    /**
     * 根据文件uuid(md5生成的)来判断此文件在服务器中是否未上传完整,
     * 如果没上传完整,则返回相关上传进度等信息
     *
     * @param pointFileIndexVo
     * @return
     */
    @Override
    public ResultEntity<PointFileIndexVo> checkUploadFileIndex(PointFileIndexVo pointFileIndexVo) {
        try {
            List<String> list = uploadProgress.get(pointFileIndexVo.getFileMd5());
            if (list == null) list = new ArrayList<>();
            pointFileIndexVo.setParts(list);
            System.out.println("已上传部分:" + list);
            return ResultEntity.success(pointFileIndexVo);
        } catch (Exception e) {
            log.error("上传文件检测异常!", e);
            return ResultEntity.error("上传文件检测异常!");
        }
    }

    /**
     * 单文件(分片)断点上传
     *
     * @param filePart 需要上传的分片文件
     * @param fileInfo 当前需要上传的分片文件信息,如uuid,文件名,文件总分片数量等
     * @return
     */
    @Override
    public ResultEntity<String> singleFilePartPointUpload(MultipartFile filePart, String fileInfo) {
        PointFileIndexVo pointFileIndexVo = JSONObject.parseObject(fileInfo, PointFileIndexVo.class);
        //实际情况下,这些路径都应该是服务器上面存储文件的路径
        String filePath = System.getProperty("user.dir") + "\\file\\";//文件存放路径
        String tempPath = filePath + "temp\\" + pointFileIndexVo.getFileMd5();//临时文件存放路径
        File dir = new File(tempPath);
        if (!dir.exists()) dir.mkdirs();

        //生成一个临时文件名
        String tempFileNamePath = tempPath + "\\" + pointFileIndexVo.getFileName() + "_" + pointFileIndexVo.getPartIndex() + ".part";
        try {
            //将分片存储到临时文件夹中
            filePart.transferTo(new File(tempFileNamePath));

            List<String> partIndex = uploadProgress.get(pointFileIndexVo.getFileMd5());
            if (Objects.isNull(partIndex)) {
                partIndex = new ArrayList<>();
            }
            partIndex.add(pointFileIndexVo.getPartIndex().toString());
            uploadProgress.put(pointFileIndexVo.getFileMd5(), partIndex);

            File tempDir = new File(tempPath);
            File[] tempFiles = tempDir.listFiles();

            one:
            if (pointFileIndexVo.getPartNum().equals(Objects.requireNonNull(tempFiles).length)) {
                //需要校验一下,表示已有异步程序正在合并了;如果是分布式这个校验可以加入redis的分布式锁来完成
                if (isMergePart.get(pointFileIndexVo.getFileMd5()) != null) {
                    break one;
                }
                isMergePart.put(pointFileIndexVo.getFileMd5(), tempFiles.length);
                System.out.println("所有分片上传完成,预计总分片:" + pointFileIndexVo.getPartNum() + "; 实际总分片:" + tempFiles.length);
                //读取分片数据,进行分片合并
                FileOutputStream fileOutputStream = new FileOutputStream(filePath + pointFileIndexVo.getFileName());
                //这里如果分片很多的情况下,可以采用多线程来执行
                for (int i = 0; i < pointFileIndexVo.getPartNum(); i++) {
                    FileInputStream fileInputStream = new FileInputStream(tempPath + "\\" + pointFileIndexVo.getFileName() + "_" + i + ".part");
                    byte[] buf = new byte[1024 * 8];//8MB
                    int length;
                    while ((length = fileInputStream.read(buf)) != -1) {//读取fis文件输入字节流里面的数据
                        fileOutputStream.write(buf, 0, length);//通过fos文件输出字节流写出去
                    }
                    fileInputStream.close();
                }
                fileOutputStream.flush();
                fileOutputStream.close();

                // 删除临时文件夹里面的分片文件 如果使用流操作且没有关闭输入流,可能导致删除失败
                for (int i = 0; i < pointFileIndexVo.getPartNum(); i++) {
                    boolean delete = new File(tempPath + "\\" + pointFileIndexVo.getFileName() + "_" + i + ".part").delete();
                    File file = new File(tempPath + "\\" + pointFileIndexVo.getFileName() + "_" + i + ".part");
                }
                //在删除对应的临时文件夹
                if (Objects.requireNonNull(tempDir.listFiles()).length == 0) {
                  tempDir.delete();
                }
                isMergePart.remove(pointFileIndexVo.getFileMd5());
                uploadProgress.remove(pointFileIndexVo.getFileMd5());
            }

        } catch (Exception e) {
            log.error("单文件分片上传失败!", e);
            return ResultEntity.error(pointFileIndexVo.getFileMd5(), "单文件分片上传失败");
        }
        //通过返回成功的分片值,来验证分片是否有丢失
        return ResultEntity.success(pointFileIndexVo.getFileMd5(), pointFileIndexVo.getPartIndex().toString());
    }

    /**
     * 获取(服务器)指定文件存储路径下所有文件MD5值
     * 实际情况下,每一个文件的md5值都是单独保存在数据库或者其他存储机制中的,
     * 不需要每次都去读取文件然后获取md5值,这样多次io读取很耗性能
     *
     * @return
     * @throws Exception
     */
    @Bean
    private List<DiskFileIndexVo> getDiskFileMd5Index() throws Exception {
        String filePath = System.getProperty("user.dir") + "\\file\\part\\";
        File saveFileDir = new File(filePath);
        if (!saveFileDir.exists()) saveFileDir.mkdirs();

        List<DiskFileIndexVo> diskFileIndexVoList = new ArrayList<>();
        File[] listFiles = saveFileDir.listFiles();
        if (listFiles == null) return diskFileIndexVoList;

        for (File listFile : listFiles) {
            String fileName = listFile.getName();
            FileInputStream fileInputStream = new FileInputStream(filePath + fileName);
            String md5DigestAsHex = DigestUtils.md5DigestAsHex(fileInputStream);

            DiskFileIndexVo diskFileIndexVo = new DiskFileIndexVo();
            diskFileIndexVo.setFileName(fileName);
            diskFileIndexVo.setFileMd5(md5DigestAsHex);
            diskFileIndexVoList.add(diskFileIndexVo);
            fileInputStream.close();
        }

        diskFileIndexVos = diskFileIndexVoList;
        log.info("服务器磁盘所有文件 {}", diskFileIndexVoList);
        return diskFileIndexVoList;
    }

 代码结构图:

五、前端代码

   整体的过程如下

前端将文件按照百分比进行计算,每次上传文件的百分之一(文件分片),给文件分片做上序号及文件uuid,然后在循环里面对文件片段上传的时候在将当前分片值一起传给后端。
后端将前端每次上传的文件,放入到缓存目录;
前端将全部的文件内容都上传完毕后,发送一个合并请求;
后端合并分片的之后对文件进行命名保存;
后端保存临时分片的时候命名索引,方便合并的时候按照分片索引进行合并;

vue模板代码:

      <!-- 单文件分片上传 -->
      <div class="fileUploadStyle">
        <h3>单文件分片上传</h3>
        <el-upload ref="upload" name="files" action="#" :on-change="selectSinglePartFile"
          :on-remove="removeSingleFilePart" :file-list="singleFilePart.fileList" :auto-upload="false">
          <el-button slot="trigger" size="small" type="primary">选取文件</el-button>
          <el-button style="margin-left: 10px;" size="small" type="success"
            @click="singleFilePartUpload">点击进行单文件分片上传</el-button>
          <div slot="tip" class="el-upload__tip">主要用于测试单文件分片上传</div>
        </el-upload>
        <el-progress :text-inside="true" class="progress" :stroke-width="26" :percentage="singlePartFileProgress" />
      </div>
      <!-- 多文件分片上传 -->
      <div class="fileUploadStyle">
        <h3>多文件分片上传</h3>
        <el-upload ref="upload" name="files" action="#" :on-change="selectMultiplePartFile"
          :on-remove="removeMultiplePartFile" :file-list="multipleFilePart.fileList" :auto-upload="false">
          <el-button slot="trigger" size="small" type="primary">选取文件</el-button>
          <el-button style="margin-left: 10px;" size="small" type="success"
            @click="multipleFilePartUpload">点击进行多文件分片上传</el-button>
          <div slot="tip" class="el-upload__tip">主要用于测试多文件分片上传</div>
        </el-upload>
      </div>
      <!-- 多文件(分片)秒传 -->
      <div class="fileUploadStyle">
        <h3>多文件(分片MD5值)秒传</h3>
        <el-upload ref="upload" name="files" action="#" :on-change="selectMultiplePartFlashFile"
          :on-remove="removeMultiplePartFlashFile" :file-list="multipleFilePartFlash.fileList" :auto-upload="false">
          <el-button slot="trigger" size="small" type="primary">选取文件</el-button>
          <el-button style="margin-left: 10px;" size="small" type="success"
            @click="multipleFilePartFlashUpload">点击进行文件秒传</el-button>
          <div slot="tip" class="el-upload__tip">主要用于测试多文件(分片MD5值)秒传</div>
        </el-upload>
      </div>

js属性定义:

上传部分代码:

minio分片上传:

上传样式:

六、功能演示及源码

部分演示图: 这里就以上传minio为例,测试上传minio以分片方式上传

以8M进行分切 28M刚好分了四个区,我们使用redis客户工具查看

最后成功上传到minio中

而且看到上传文件大小为:28M

       文件上传代码其实功能也简单也很明确,先将一个大文件分成n个小文件,然后给后端检测这些分片是否曾经上传中断过,即对这些分片进行过滤出来,并将过滤出对应的分片定位值结果返回给前端处理出不需要上传的分片和需要上传的文件分片,这里主要还是区分到确定是这个文件的分区文件。

        这里,为了方便大家直接能够使用Java源码,本文所有都采用Spring boot框架模式,另外使用了第三方插件,如果大家使用中没有使用到minio可以不需要引入并把相关代码移除即可,代码使用了redis作为分区数量缓存,相对于Java内存更稳定些。

demo源码下载gitee地址(代码包含Java后端工程和vue2前端工程):java-file-upload-demo: java 多文件上传、多文件分片上传、多文件秒传、minio分片上传等功能

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/672878.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

nginx的配置粗记

小白nginx的配置随笔&#xff08;随便记记&#xff09; 前言 我们都知道nginx有很多用途&#xff0c;比如&#xff1a;负载均衡&#xff0c;反向代理&#xff0c;网关路由&#xff0c;解决跨域等问题。我这次开发项目&#xff0c;用到的一些功能也涉及到了对nginx的配置&#…

Spark介绍及RDD操作

Spark介绍及RDD操作 PySpark简介spark特点运行原理spark实例化 SparkCore-RDDRDD创建转换&#xff08;Transformation&#xff09;行动&#xff08;Action&#xff09; PySpark简介 spark特点 运行速度快&#xff1a;DAG内存运算容易使用&#xff1a;Java、Scala、Python、R通…

Kubernetes——YAML文件编写

目录 一、创建Kubernetes对象YAML文件必备字段 1.apiVersion 2.kind 3.metadata 4.spec 二、YAML格式基本规范 1.结构表示 2.键值对 3.列表&#xff08;数组&#xff09; 4.字典&#xff08;映射&#xff09; 5.数据类型 6.注释 7.多文档支持 8.复杂结构 9.示例 …

快速排序与归并排序(非递归)

目录 快速排序&#xff08;双指针法&#xff09; 原理 代码 快速排序&#xff08;非递归&#xff09; 原理 代码 归并排序 介绍 优点 缺点 图片 原理 代码 归并排序&#xff08;非递归&#xff09; 代码 快速排序&#xff08;双指针法&#xff09; 快速排序的精…

【讲解下常见的分类算法,什么是分类算法?】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

redis如何实现分布式锁

Redisson是怎么实现分布式锁的 分布式锁&#xff1a;Redisson 提供了一种简单而强大的方式来实现分布式锁。 它支持多种锁模式&#xff0c;如公平锁、可重入锁、读写锁等&#xff0c;并且提供了锁的超时设置和自动释放功能。 锁的获取 在Redisson中常见获取锁的方式有 lock() …

面向对象概述

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 面向对象&#xff08;Object Oriented&#xff09;的英文缩写是OO&#xff0c;它是一种设计思想。从20世纪60年代提出面向对象的概念到现在&#xff…

Cyber Weekly #9

赛博新闻 1、OpenAI&#xff1a;GPTs向全部用户开放&#xff0c;使用GPT-4o OpenAI宣布所有ChatGPT免费用户现在可以在GPT商店中使用GPTs&#xff0c;并且这些GPTs现在使用最新的GPT-4o模型。 2、马斯克 vs. Yann LeCun 这一周&#xff0c;AI圈最热闹的莫过于马斯克和LeCun的…

scGPT实验解读

本篇内容为发表在Nature Methods上的scGPT的部分实验内容 来自&#xff1a;scGPT: toward building a foundation model for single-cell multi-omics using generative AI, Nature Methods, 2024 目录 scGPT揭示特定细胞状态的基因网络缩放法则和迁移学习中的上下文效应 scGP…

基于安卓的虫害识别软件设计--(2)模型性能可视化|混淆矩阵、热力图

1.混淆矩阵&#xff08;Confusion Matrix&#xff09; 1.1基础理论 &#xff08;1&#xff09;在机器学习、深度学习领域中&#xff0c;混淆矩阵常用于监督学习&#xff0c;匹配矩阵常用于无监督学习。主要用来比较分类结果和实际预测值。 &#xff08;2&#xff09;图中表达…

物理模拟技术在AI绘画中的革新作用

引言&#xff1a; 随着人工智能&#xff08;AI&#xff09;技术的飞速发展&#xff0c;艺术领域也迎来了一场创新的革命。AI绘画&#xff0c;作为这场革命的重要组成部分&#xff0c;不仅改变了传统艺术创作的模式&#xff0c;而且为艺术家提供了前所未有的创作工具。在这一过程…

Linux基础1-基本指令1

1.Linux学习前言 Linux的学习非常重要&#xff0c;我们学习Linux的第一步是在电脑中搭建Linux环境。 对于没有搭建过的可以看这阿伟t的一篇文章 【Linux入门】Linux环境配置-CSDN博客 我的环境为XShell,运行的云服务器是阿里云 2.本章重点 1.显示当前目录下的所有文件…

软件杯 题目:基于卷积神经网络的手写字符识别 - 深度学习

文章目录 0 前言1 简介2 LeNet-5 模型的介绍2.1 结构解析2.2 C1层2.3 S2层S2层和C3层连接 2.4 F6与C5层 3 写数字识别算法模型的构建3.1 输入层设计3.2 激活函数的选取3.3 卷积层设计3.4 降采样层3.5 输出层设计 4 网络模型的总体结构5 部分实现代码6 在线手写识别7 最后 0 前言…

展现市场布局雄心,ATFX再度亮相非洲峰会,开启区域市场新篇章

自2023年全球市场营销战略部署实施以来&#xff0c;ATFX在全球各区域市场取得了丰硕成果&#xff0c;其品牌实力、知名度、影响力均有大幅提升。在这场全球扩张的征程中&#xff0c;非洲市场日益成为集团关注的焦点。自2023年首次踏上这片充满潜力的市场以来&#xff0c;ATFX持…

定义类并创建类的实例

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 在Python中&#xff0c;类表示具有相同属性和方法的对象的集合。在使用类时&#xff0c;需要先定义类&#xff0c;然后再创建类的实例&#xff0c;通…

谨以此文章记录我的蓝桥杯备赛过程

以国优秀结束了蓝桥杯cb组 鄙人来自电信学院&#xff0c;非科班出身&#xff0c;在寒假&#xff0c;大约2024年2月份&#xff0c;跟着黑马程序员将c基础语法学完了&#xff0c;因为过年&#xff0c;事情较多&#xff0c;没在学了。 最初就是抱着拿省三的态度去打这个比赛的&a…

低代码是什么?开发系统更有什么优势?

低代码&#xff08;Low-Code&#xff09;是一种应用开发方法&#xff0c;它采用图形化界面和预构建的模块&#xff0c;使得开发者能够通过少量的手动编程来快速创建应用程序。这种方法显著减少了传统软件开发中的手动编码量&#xff0c;提高了开发效率&#xff0c;降低了技术门…

图形学初识--多边形剪裁算法

文章目录 前言正文为什么需要多边形剪裁算法&#xff1f;前置知识二维直线直线方程&#xff1a;距离本质&#xff1a;点和直线距离关系&#xff1a; 三维平面平面方程距离本质&#xff1a;点和直线距离关系&#xff1a; Suntherland hodgman算法基本介绍基本思想二维举例问题描…

mysql中EXPLAIN详解

大家好。众所周知&#xff0c;MySQL 查询优化器的各种基于成本和规则的优化会后生成一个所谓的执行计划&#xff0c;这个执行计划展示了接下来具体执行查询的方式。在日常工作过程中&#xff0c;我们可以使用EXPLAIN语句来查看某个查询语句的具体执行计划&#xff0c; 今天我们…

椭圆轨道的周期性运动轨道

一、背景介绍 本节将从轨道六根数的角度&#xff0c;探究目标星为椭圆轨道&#xff0c;追踪星周期性环绕目标的必要条件。根据航天动力学的原理&#xff0c;对于一个椭圆轨道&#xff0c;其轨道能量为 对于能够不产生漂移的情况&#xff0c;绕飞编队的能量。对于追踪星到目标星…