评论

收藏

[Java] Java实现文件切割拼接的实现代码

编程语言 编程语言 发布于:2021-10-07 13:03 | 阅读数:320 | 评论:0

这篇文章主要介绍了Java实现文件切割拼接的实现代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
单线程实现
文件分割
在老的fat32文件系统中,最大的单个文件大小必须保存在4g内,对于经常看电影的我这个是不能允许的。不过现在windows有ntfs文件系统,linux大部分发行版为ext4文件系统,最大单个文件大小能大于4g。不过这二者并不能兼容。。格式化ntfs的u盘linux不能识别,格式化ext4的u盘windows不能识别,只能用老的fat32兼容二者。所以将文件分割,再进行拼接就很重要,文件经过分割了在网络上传输就十分方便,也能开多线程对每部分进行hash提高处理效率。
DSC0000.png

最近看的bradpitt的《狂怒》
首先:对文件进行分割需要确定每一部分的大小,假设上面的 fury.mkv 文件大小为 280m ,分割每一块设置默认大小为 64m ,所以:
DSC0001.png

对于最后一块,一般小于等于设定好的每块默认大小。 每块大小设置好了,接下来,就需要将文件的路径获取,代码中搭建输入流,将文件读入内存缓冲区中,再搭建输出流,将缓冲区输出到新的分割文件中。 再接下来实现就很简单了。 新建一个 fileslice
类:有切割方法,拼接方法。
public class fileslice {
 /**
  * 分割文件
  * @param filepath 文件路径
  * @param filepiecesize 文件每块大小,单位为字节,为-1则默认为每块64m
  * @return 成功返回true,出错则返回false
  */
 public static boolean slice(path filepath, int filepiecesize){
  return true;
 }
 
 /**
  * 将分割好的文件重新链接
  * @param filepath 被分割好的其中之一文件路径,默认其他块与其在同一目录下
  * @param howmanyparts 一共有多少块
  * @return 成功返回true,出错则返回false
  */
 public static boolean glue(path filepath, int howmanyparts){
  return true;
 }
}
接下来实现单线程的分割方法: 用图解的话应该是这样:
DSC0002.png

代码实现: 进入函数首先判断文件是否存在:
if (!files.exists(filepath)){
 return false;
}
接下来判断每块大小是否使用默认值:
if(filepiecesize == -1){
 filepiecesize = 1024*1024*64;
}
将路径转换为文件对象,再计算将分割多少块:
file file = filepath.tofile();
int howmanyparts = (int) math.ceil(file.length() / (double)filepiecesize);
初始化输入输出流,出错输出错误信息,返回false,获得当前目录:
datainputstream filereader = null;
try {
 filereader = new datainputstream(new fileinputstream(file));
} catch (filenotfoundexception e) {
 e.printstacktrace();
 system.out.println("文件找不到!");
 return false;
}
dataoutputstream filewriter;
path dir = filepath.getparent();
接下来读取文件,并且分别输出到各个part文件中:
int readlength = -1;
long total = 0;
 
try {
 for (int i = 1; i <= howmanyparts ; i++){
  //新建文件part i
  path temp = files.createfile(dir.resolve(filepath.getfilename() + ".part" + i));
  //搭建输出流
  filewriter = new dataoutputstream(new fileoutputstream(temp.tofile()));
  //读取文件并输出
  while ( (readlength = filereader.read(buffer)) != -1){
   filewriter.write(buffer,0,readlength);
   filewriter.flush();
   total += readlength;
   if (total == filepiecesize){
  total = 0;
  break;
   }
  }
  //part i的文件已经输出完毕,关闭流
  filewriter.close();
 }
 //读取完毕,关闭输入流
 filereader.close();
} catch (ioexception e) {
 e.printstacktrace();
 system.out.println("io错误!");
 return false;
}
该函数已经实现完毕,接下来测试(由于电影fury有14g。。太大了。。还是换个吧):
DSC0003.jpg

我是大哥大第5集,有729m,大概能分个12个part吧。
public static void main(string[] args) throws ioexception {
 double before = system.currenttimemillis();
 
 path bigboss = paths.get("d:\\video\\我是大哥大\\我是大哥大.kyou.kara.ore.wa.ep05.chi_jap.hdtvrip.1280x720.mp4");
 
 fileslice.slice(bigboss,-1);
 
 double after = system.currenttimemillis();
 
 system.out.println("分割文件我是大哥大.kyou.kara.ore.wa.ep05.chi_jap.hdtvrip.1280x720.mp4," + files.size(bigboss) + "字节,总用时" + (after - before) + "ms" );
 
}
运行结果:
分割文件我是大哥大.kyou.kara.ore.wa.ep05.chi_jap.hdtvrip.1280x720.mp4,765321889字节,总用时16335.0ms
DSC0004.jpg

速度还是挺慢的。。 下次还是换成多线程来实现,再来测试下速度。在单线程情况下一个普通的40分钟日剧都要15-30s左右,要是mkv格式的电影都要好久了。。不过其实极限应该不在cpu中执行的速度,而是在硬盘io中,如果是普通硬盘那么就算是多线程也应该提速不了多少。。
文件拼接
这个就很简单了,和分割相反就ok。 直接上完整代码:
public static boolean glue(path filepath, int howmanyparts){
 if (!files.exists(filepath)){
  return false;
 }
 //获取原始文件名
 string filename = getoriginalfilename(filepath.getfilename().tostring());
 
 if (filename == null){
  system.out.println("传入part文件名解析出错!");
  return false;
 }
 //初始化缓冲区
 byte [] buffer = new byte[1024 * 8];
 //获取文件存储的路径
 path dir = filepath.getparent();
 
 try {
  datainputstream filereader = null;
  //创建原始文件
  files.createfile(dir.resolve(filename));
  //搭建原始文件输出流
  dataoutputstream filewriter = new dataoutputstream(new fileoutputstream(dir.resolve(filename).tofile()));
 
  int readlength = -1;
  for (int i = 1; i <= howmanyparts ; i++){
   //得到part i文件路径
   path temp = dir.resolve(filename + ".part" + i);
   //搭建输入流
   filereader = new datainputstream(new fileinputstream(temp.tofile()));
   //读取文件并输出
   while ( (readlength = filereader.read(buffer)) != -1){
  filewriter.write(buffer,0,readlength);
  filewriter.flush();
   }
   //part i的文件已经读入完毕,关闭流
   filereader.close();
  }
  //写入完毕,关闭输出流
  filewriter.close();
 } catch (ioexception e) {
  e.printstacktrace();
  system.out.println("io错误!");
  return false;
 }
 return true;
}
再测试刚刚分割好的我是大哥大第5集
public static void main(string[] args) throws ioexception {
 double before = system.currenttimemillis();
 
 path bigboss = paths.get("d:\\video\\我是大哥大\\我是大哥大.kyou.kara.ore.wa.ep05.chi_jap.hdtvrip.1280x720.mp4.part1");
 
 fileslice.glue(bigboss,12);
 
 double after = system.currenttimemillis();
 
 system.out.println("拼接12个part,用时" + (after - before) + "ms");
 
}
结果输出,用12s左右,还行。
拼接12个part,用时12147.0ms
打开播放毫无问题,最后截张图。

未完待续。。下次来使用多线程进行实现。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持CodeAE代码之家
原文链接:https://juejin.im/post/5bf3f63c6fb9a049f570c158

关注下面的标签,发现更多相似文章