vue断点续传二


vue+spring 断点续传(二)

本文主要讲述断点续传的后端实现,后端实现的要点是分片的控制和并发的考虑


提示:以下是本篇文章正文内容,下面案例可供参考

一、环境

spring 2.5.0

二、后端实现

1.bean对象

FileInfo对象主要用于记录文件的信息,getter和setter请自行实现

public class FileInfo {
    private String hashCode;
    private String fileName;
    private int fileSize;
    private int totalIndex;
    private  Set<Integer> hasUploadShard = new HashSet<>();
    private Set<Integer> noUploadShard = new HashSet<>();
}

shardInfo对象主要用于记录分片的信息,getter和setter请自行实现

public class ShardInfo {
    /**
     * 文件hash值
     */
    private String hashCode;
    /**
     * 分片索引
     */
    private Integer shardIndex;
    /**
    * 分片的二进制数据
    */
    private MultipartFile shard;
}

2.逻辑实现

代码如下(示例):逻辑代码仅可作为参考,还有很多不足之处,比如文件和分片的信息未进行落库、长时间未合并的分片的处理逻辑还未实现、IO合并方式的可靠性还未进行验证…

@RestController
@CrossOrigin(origins = "*")
public class upload {
    private static final String savePath = "C:\\Users\\admin\\Desktop\\qianduan";
    private static Map<String, FileInfo> fileMap = new ConcurrentHashMap<>();

    @PostMapping("file/upload/shard")
    public Map<String, Object> fileUploadShard(MultipartFile shard, ShardInfo shardInfo) throws Exception {
        HashMap<String, Object> map = new HashMap<>();
        shardInfo.setShard(shard);
        System.out.println(shardInfo.getShardIndex());
        try {

            //  判断内存中是否已有文件,如若没有抛出异常
            FileInfo fileInfo = checkMemoryAndUpdate(shardInfo);

            //  判断分片是否已经创建
            boolean contains = fileInfo.getHasUploadShard().contains(shardInfo.getShardIndex());
            if (contains) {
                map.put("status", "分片已创建");
                map.put("status", 0);
                return map;
            }

            //  创建分片
            createShardAndUpdate(fileInfo, shardInfo);

            //  封装结果
            //  0:分片正常上传
            map.put("status", 0);
            map.put("fileInfo", fileInfo);
            //  判断并合并分片
            mergeShard(fileInfo);
            return map;
        } catch (Exception e) {
            //  1:分片上传出现异常
            map.put("status", 1);
            return map;
        }
    }

    @PostMapping("file/status")
    public Map<String, Object> getFileStatus(FileInfo fileInfo) {
        HashMap<String, Object> res = new HashMap<>();
        FileInfo info = fileMap.get(fileInfo.getHashCode());
        if (info == null) {
            //  文件首次上传
            info = new FileInfo();
            info.setFileName(fileInfo.getFileName());
            info.setFileSize(fileInfo.getFileSize());
            info.setHashCode(fileInfo.getHashCode());
            info.setTotalIndex(fileInfo.getTotalIndex());
            HashSet<Integer> set = new HashSet<>();
            for (int i = 0; i < info.getTotalIndex(); i++) {
                set.add(i);
            }
            info.setNoUploadShard(set);
            fileMap.put(info.getHashCode(), info);
        }
        //  文件有记录
        res.put("fileInfo", info);

        return res;
    }

    /**
     * 获取内存中文件信息
     *
     * @return
     */
    @GetMapping("file/list")
    public Map<String, FileInfo> getFileList() {
        return fileMap;
    }

    //  判断并合并分片
    private void mergeShard(FileInfo fileInfo) throws IOException {
        if (fileInfo.getNoUploadShard().size() == 0) {
            //  分片文件名
            ArrayList<String> shardNames = new ArrayList<>();
            //  创建文件名和文件
            String fileName = fileInfo.getHashCode() + getExtension(fileInfo.getFileName());
            File overAllFile = new File(savePath + "\\" + fileName);
            //  分片文件名
            for (int i = 0; i < fileInfo.getTotalIndex(); i++) {
                String shardName = fileInfo.getHashCode() + "_" + i + ".blob";
                shardNames.add(shardName);
            }

            //  获取输出流
            FileOutputStream out = new FileOutputStream(overAllFile);
            byte[] buf = new byte[1024];

            //  循环copy
            shardNames.forEach(val -> {
                File shardFile = new File(savePath + "\\" + val);
                int len = 0;
                try {
                    FileInputStream in = new FileInputStream(shardFile);
                    while ((len = in.read(buf)) != -1) {
                        out.write(buf, 0, len);
                    }
                    in.close();
                    out.flush();
                } catch (Exception e) {
                    overAllFile.delete();
                    e.printStackTrace();
                }
            });
            //  关闭
            out.close();
            //  成功后删除分片
            shardNames.forEach(val -> {
                File shardFile = new File(savePath + "\\" + val);
                shardFile.delete();
            });

        }
    }

    /**
     * 获取文件后缀
     *
     * @param fileName
     * @return
     */
    private String getExtension(String fileName) {
        if (fileName == null) {
            return null;
        }
        int extensionPos = fileName.lastIndexOf(46);
        int lastUnixPos = fileName.lastIndexOf(47);
        int lastWindowsPos = fileName.lastIndexOf(92);
        int lastSeparator = Math.max(lastUnixPos, lastWindowsPos);
        if (lastSeparator < extensionPos) {
            //
            return "." + fileName.substring(extensionPos + 1);
        } else {
            return "";
        }

    }


    //  创建分片并更新信息
    private void createShardAndUpdate(FileInfo fileInfo, ShardInfo shardInfo) throws IOException {
        File file1 = new File(savePath + "\\" + fileInfo.getHashCode() + "_" + shardInfo.getShardIndex() + ".blob");
        FileCopyUtils.copy(shardInfo.getShard().getInputStream(), new FileOutputStream(file1));
        //  更新分片数据
        updateUploadShard(fileInfo, shardInfo.getShardIndex());
    }

    //  检查内存是否有文件记录,没有则添加
    public FileInfo checkMemoryAndUpdate(ShardInfo fileShard) throws Exception {
        FileInfo info = fileMap.get(fileShard.getHashCode());
        if (info == null) {
            throw new Exception("文件不存在");
        }
        return info;
    }

    private void updateUploadShard(FileInfo fileInfo, Integer shardIndex) {
        //  添加已经上传的shardIndex
        fileInfo.getHasUploadShard().add(shardIndex);
        //  更新未上传的shardIndex
        setNoUploadShard(fileInfo);

    }

    private void setNoUploadShard(FileInfo fileInfo) {
        Set<Integer> uploadShard = fileInfo.getHasUploadShard();
        Set<Integer> integers = new HashSet<>();
        for (int i = 0; i < fileInfo.getTotalIndex(); i++) {
            boolean contains = uploadShard.contains(i);
            //  如若不含
            if (!contains) {
                integers.add(i);
            }
        }
        fileInfo.setNoUploadShard(integers);
    }

}

总结

加油噢!

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值