大文件分片上传的后台方法(Java)

本文仅为后端Java,如需前端Vue,请移步我的另一篇文章:支持IE11的文件分片逐一上传的方法(vue)

1 数据表结构

file表

字段名数据类型主键非空自增注释
mdvarchar++-文件唯一标识码
allNumint---文件全部分片数量
successNumint---已经上传成功的分片数量
typeint---文件状态
pathvarcher---文件在服务器磁盘路径

file_info表

字段名数据类型主键非空自增注释
idint+++varchar
mdvarcher---文件唯一标识码
fileNamevarchar---文件真是名字
detailvarchar---详细介绍
datevarchar---文件上传时
fileTypeIdint---文件类型
parentIdint---文件类型父级ID

2 Java后台

Bean层

@Data
@Accessors(chain = true)
public class FileInfo{
	private Integer id;
	private String md;
	private Integer allNum;
	private Integer successNum;
	private String fileName;
	private String detail;
	private String path;
	private String size;
	private String date;
	private Integer fileTypeId;
	private Integer parentId;
}

2.1 工具类

工具类Tools,下边两个方法在这个工具类中

@Component("Tools")
public class Tools{

}

将前端传来的文件分片存储到服务器的磁盘中,磁盘的IO操作,没什么需要注释的吧

/** 
* 将流文件保存至指定磁盘位置
* @Param name 文件的唯一标识码(MD)
* @Param chunkPath 分片文件需要保存到的服务器磁盘路径
* @Param file 分片文件的流对象
* @Param num 本次上传的是分片文件的第几个分片
* @return boolean 返回值确认是否操作成功
*/
public boolean saveFileChunk(String name,String chunkPath,MultipartFile file,Integer num){
	OutputStream out = null;
	InputStream in = null;
	try{
		in = file.getInputStream();
	}catch(Exception e){
		e.printStackTrace;
	}
	try{
		byte[] bs = new byte[1024];
		int len;
		File fileChunk = new File(chunkPath);
		if(!fileChunk.exists()){
			fileChunk.mkdirs();
		}
		out = new FileOutputStream(chunkPath + "\\" + name + "_" + num + ".temporary");
		while((len=in.read(bs)) != -1){
			out.write(bs,0,len);
		}
	}catch(Exception e){
		e.printStackTrace();
		return false;
	}finally{
		try{
			out.close();
			in.close();
		}catch(Exception e){
			e.printStackTrace();
		}
	}
	return true;
}

将指定文件夹、文件名的分片文件合并为完整文件,同时删除分片文件

/** 
* 将指定文件夹、文件名的分片文件合并为完整文件,同时删除分片文件
* @Param fileName 文件唯一标识码(md)
* @Param path 分片文件所在磁盘路径
* @param num 总分片数量
* @return boolean 确定操作是否成功
*/
public Boolean mergeFileChunk(String fileName,String path,Integer num){
	OutputStream = null;
	InputStream = null;
	File fileChunk = null;
	try{
		new File(path + "\\" + fileName).delete();
		out = new BufferedOutputStream(new FileOutputStream(path + "\\" + fileName,true));
		for(int i = 0 ; i < num ; i ++){
			fileChunk = new file(path + "\\" + fileName + "_" + i + ".temporary");
			if(!fileChunk.exists()){
				continue;
			}
			in = new BufferedInputStream(new FileInputStream(fileChunk));
			byts[] chunk = new byte[1024];
			int theEnd = -1;
			while((theEnd = in.read(chunk)) != -1){
				out.write(chunk,0,theEnd);
			}
			in.close();
			fileChunk.delete();
		}
	}catch(Exception e){
		e.printStackTrace();
		return false;
	}final{
		try{
			out.close();
		}catch(Excepting e){
			e.printStackTrace();
		}
	}
	return true;
}

ServiceImpl层的一个公用方法

private FileInfo getFileInfo(HttpServletRequest info){
	fileInfo fileInfo = new FileInfo();
	fileInfo.setMd(info.getParameter("md"))
			.setAllNum(Integer.valueOf(info.getParameter("allNum")))
			.setSuccessNum(Integer.valueOf(info.getParameter("successNum")))
			.setType(Integer.valueOf(info.getParameter("type")))
			.setFileName(info.getParameter("fileIName"))
			.setDetail(info.getParameter("detail"))
			.setPath(info.getParameter("path"));
}

2.2 实现查询文件状态

很简单的查询,可以忽略不看
Controller层

public Result findFileInfo(@RequestBody FileInfo fileInfo){
	return new Result(fileService.findFileInfo(fileInfo))
}

ServiceImpl层

public List<FileInfo> findFileInfo(FileInfo fileInfo){
	return new Result(fileService.findFileInfo(fileInfo))
}

Mapper层

@Select("select a.md md,a.allNum allNum,a.successNum successNum,a.type type,a.path path,b.id id,b.fileName fileName,b.detail detail from file a left join file_info b on a.md=b.md where a.md=#{md}")
List<FileInfo> findFileInfo(FileInfo fileInfo);

2.3 实现首次上传文件分片

Controller层

@PostMapping("/uploadFileFirst")
public Result uploadFileFirst(HttpServletRequest httpServletRequest,
									@RequestParam("file") MultopartFile multipartFile){
	return new Result(fileService.uploadFileFirst(httpServletRequest,multopartFile))
}

ServiceImpl层

public Result uploadFileFirst(HttpServletRequest info,MultopartFile file){
	FileInfo fileInfo = getFileInfo(info);	//2.1工具类中有这个方法
	//这里是获取文件类型的数据,为了可以分类型查找文件,不是文件上传的必要参数
	fileInfo.setParentId(Integer.valueOf(info.getParameter("parentId")))
			.setFileTypeId(Integer.valueOf(info.getParameter("fileTypeId")));
	String[] paths = new simpleDateFormat("yy.MM.dd-HH:mm:ss".format(new Date())).split("-")[0].split("\\.");
	//这里需要自定义文件存储根目录位置
	fileInfo.setPath("........." + "\\" + path[0] + "\\" + path[1] + "\\" +path[2] + "\\" + fileInfo.getMd())
			.setDate(newsimpleDateFormat("yy.MM.dd-HH:mm:ss".format(new Date())))
			.setType(0);
	fileMapper.insertFileInfo(fileInfo);
	fileMapper.insertNewFileInfo(fileInfo);
	tools.saveFileChunk(fileInfo.getMd(),fileInfo.getPath(),file,fileInfo.getSuccessNum());
	fileMapper.updateFileSuccessNum(fileInfo.getMd(),fileInfo.getSuccessNum()+1);
	fileInfo.setSuccessNum(fileInfo.getSuccessNum+1).setType(0);
	if(fileInfo.getAllNum() == fileInfo.getSuccessNum()&&fileInfo.getAllNum == 1){
		fileInfo.setType(1);
		fileMapper.updateFileType(fileInfo.getMd(),fileInfo.getType());
		tools.mergeFileChunk(fileInfo.getMd(),fileInfo.getPath(),fileInfo.getAllNum());
	}
	return fileInfo;
}

Mapper层

@Insert("insert into file_info values(null,#{md}.#{fileName},#{detail},#{date},#{fileTypeId},#{parentId})")
void insertFileInfo(FileInfo fileInfo);

@Insert("insert into file values(#{md}.#{allNum},#{successNum},#{type},#{path})")
void insertNewFileInfo(FileInfo fileInfo);

@Update("update file set successNum=#{successNum} where md=#{md}")
void updateFileSuccessNum(@Param("md")String md,@Param("successNum")Integer successNum);

@Update("update file set type=#{type} where md=#{md}")
void updateFileType(@Param("md")String md,@Param("type")Integer type);

2.4 实现继续上传文件分片

Controller层

@PostMapping("/uploadFileContinue")
public Result uploadFileContinue(HttpServletRequest httpServletRequest,
									@@RequestParam("file") MultopartFile multopartFile){
	return new Result(fileService.uploadFileContinue(httpServletRequest,multopartFile))
}

ServiceImpl层

public Result uploadFileContinue(HttpServletRequest info,MultopartFile file){
	FileInfo fileInfo = getFileInfo(info);	//2.1工具类中有这个方法
	tools.saveFileChunk(fileInfo.getMd(),fileInfo.getPath(),file,fileInfo.getSuccessNum());
	fileMapper.updateFileSuccessNum(fileInfo.getMd(),fileInfo.getSuccessNum()+1);
	fileInfo.setSuccessNum(fileInfo.getSuccessNum+1);
	if(fileInfo.getAllNum() == fileInfo.getSuccessNum()&&fileInfo.getAllNum == 1){
		fileInfo.setType(1);
		fileMapper.updateFileType(fileInfo.getMd(),fileInfo.getType());
		tools.mergeFileChunk(fileInfo.getMd(),fileInfo.getPath(),fileInfo.getAllNum());
	}
	return fileInfo;
}

Mapper层

@Update("update file set successNum=#{successNum} where md=#{md}")
void updateFileSuccessNum(@Param("md")String md,@Param("successNum")Integer successNum);

@Update("update file set type=#{type} where md=#{md}")
void updateFileType(@Param("md")String md,@Param("type")Integer type);

2.5 实现秒传

Controller层

@PostMapping("/uploadFileFast")
public Result uploadFileFast(@RequestBody FileInfo fileInfo){
	return new Result(fileService.uploadFileFast(fileInfo))
}

ServiceImpl层

public Result uploadFileFast(FileInfo fileInfo){
	fileInfo.steType(1).setSuccessNum(fileInfo.getAllNum()).setDate(new simpleDateFormat("yy.MM.dd-HH:mm:ss".format(new Date())));
	fileMapper.insertFileInfo(fileInfo);
	return fileInfo;
}

Mapper层

@Insert("insert into file_info values(null,#{md}.#{fileName},#{detail},#{date},#{fileTypeId},#{parentId})")
void insertFileInfo(FileInfo fileInfo);

本文全部更新已完成

如果觉得不错请点赞收藏支持一下,有问题请评论

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值