14

我有一个奇怪的问题要问 - 如何使用 Java(任何文件类型)将给定文件分成小块?然后我可以把这些碎片放在一些 CD 和/或随身碟上,然后把它们拿走。我试过这样

但正如大多数用户所评论的那样,我正在尝试的事情无法以这种方式实现。因此,我决定提出一个新问题,以获得正确的文件破解方法。

当我将一个文件分成几块(比如 30 块)时,必须有一种方法可以重新组合它们并重新创建原始文件。请帮忙。

4

2 回答 2

43

当缓冲区已满或文件结束时,将部分字节读取到字节数组并将它们存储在新文件中。

例如(代码并不完美,但应该有助于理解流程)

class FileSplit {
    public static void splitFile(File f) throws IOException {
        int partCounter = 1;//I like to name parts from 001, 002, 003, ...
                            //you can change it to 0 if you want 000, 001, ...

        int sizeOfFiles = 1024 * 1024;// 1MB
        byte[] buffer = new byte[sizeOfFiles];

        String fileName = f.getName();

        //try-with-resources to ensure closing stream
        try (FileInputStream fis = new FileInputStream(f);
             BufferedInputStream bis = new BufferedInputStream(fis)) {

            int bytesAmount = 0;
            while ((bytesAmount = bis.read(buffer)) > 0) {
                //write each chunk of data into separate file with different number in name
                String filePartName = String.format("%s.%03d", fileName, partCounter++);
                File newFile = new File(f.getParent(), filePartName);
                try (FileOutputStream out = new FileOutputStream(newFile)) {
                    out.write(buffer, 0, bytesAmount);
                }
            }
        }
    }

    public static void main(String[] args) throws IOException {
        splitFile(new File("D:\\destination\\myFile.mp4"));
    }
}

myFile.mp4 大小=12,7 MB

拆分后我有13个文件

  • myFile.mp4.001-myFile.mp4.012大小为 1 MB
  • myFile.mp4.013大小为 806 KB

如果要合并这些文件,可以使用

public static void mergeFiles(List<File> files, File into)
        throws IOException {
    try (FileOutputStream fos = new FileOutputStream(into);
         BufferedOutputStream mergingStream = new BufferedOutputStream(fos)) {
        for (File f : files) {
            Files.copy(f.toPath(), mergingStream);
        }
    }
}

您还可以创建一些额外的方法来让您的生活更轻松。例如,该方法将根据其中一个文件的名称(和位置)创建包含分隔部分的文件列表。

public static List<File> listOfFilesToMerge(File oneOfFiles) {
    String tmpName = oneOfFiles.getName();//{name}.{number}
    String destFileName = tmpName.substring(0, tmpName.lastIndexOf('.'));//remove .{number}
    File[] files = oneOfFiles.getParentFile().listFiles(
            (File dir, String name) -> name.matches(destFileName + "[.]\\d+"));
    Arrays.sort(files);//ensuring order 001, 002, ..., 010, ...
    return Arrays.asList(files);
}

使用该方法,我们可以重载mergeFiles方法以仅使用其中一个文件File oneOfFiles而不是整个列表List<File>(我们将根据其中一个文件生成该列表)

public static void mergeFiles(File oneOfFiles, File into)
        throws IOException {
    mergeFiles(listOfFilesToMerge(oneOfFiles), into);
}

你也可以重载这些方法来String代替File(我们将在需要时将每个字符串包装在 File 中)

public static List<File> listOfFilesToMerge(String oneOfFiles) {
    return listOfFilesToMerge(new File(oneOfFiles));
}

public static void mergeFiles(String oneOfFiles, String into) throws IOException{
    mergeFiles(new File(oneOfFiles), new File(into));
}
于 2012-06-02T17:12:19.107 回答
3

以 raw 形式读取和写入流byte[]。避免文本流和读者。

在您的最后一个问题中,您显然是根据“行”分解文件。要复制该行为,只需使用固定大小的byte[]来读取。仔细注意上一个问题的注释中的警告,以检查读取了多少字节。Abyte[2^16]不一定会在一次读取中填写。

于 2012-06-02T17:10:33.670 回答