Добавить строки в файл в Java - PullRequest
20 голосов
/ 29 марта 2010

Есть ли способ добавить строку в файл в Java, не создавая временный файл и не записав в него необходимый контент?

Ответы [ 6 ]

20 голосов
/ 29 марта 2010

Нет, нет способа сделать это БЕЗОПАСНО в Java.

Ни одна реализация файловой системы ни в одной основной операционной системе не поддерживает подобные вещи, и вы не найдете эту функцию поддерживаемой ни в одном из основных языков программирования.

Реальные файловые системы реализованы на устройствах, которые хранят данные в виде «блоков» фиксированного размера. Невозможно реализовать модель файловой системы, в которой вы можете вставлять байты в середину файла без существенного замедления файлового ввода-вывода, траты дискового пространства или обоих.


Решения, предусматривающие перезапись файла на месте, изначально небезопасны. Если ваше приложение будет убито или питание прекратится в процессе подготовки / перезаписи, вы, скорее всего, потеряете данные. Я бы НЕ рекомендовал использовать этот подход на практике.

Использовать временный файл. Это безопаснее.

6 голосов
/ 29 марта 2010

Существует способ, который предполагает перезапись всего файла (но без временного файла). Как уже упоминалось, ни одна файловая система не поддерживает добавление содержимого в файл. Вот пример кода, который использует RandomAccessFile для записи и чтения содержимого, сохраняя при этом часть содержимого в буфере:

public static void main(final String args[]) throws Exception {
    File f = File.createTempFile(Main.class.getName(), "tmp");
    f.deleteOnExit();

    System.out.println(f.getPath());

    // put some dummy content into our file
    BufferedWriter w = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(f)));
    for (int i = 0; i < 1000; i++) {
        w.write(UUID.randomUUID().toString());
        w.write('\n');
    }
    w.flush();
    w.close();

            // append "some uuids" to our file
    int bufLength = 4096;
    byte[] appendBuf = "some uuids\n".getBytes();
    byte[] writeBuf = appendBuf;
    byte[] readBuf = new byte[bufLength];

    int writeBytes = writeBuf.length;

    RandomAccessFile rw = new RandomAccessFile(f, "rw");
    int read = 0;
    int write = 0;

    while (true) {
                    // seek to read position and read content into read buffer
        rw.seek(read);
        int bytesRead = rw.read(readBuf, 0, readBuf.length);

                    // seek to write position and write content from write buffer
        rw.seek(write);
        rw.write(writeBuf, 0, writeBytes);

                    // no bytes read - end of file reached
        if (bytesRead < 0) {
                            // end of
            break;
        }

                    // update seek positions for write and read
        read += bytesRead;
        write += writeBytes;
        writeBytes = bytesRead;

                    // reuse buffer, create new one to replace (short) append buf
        byte[] nextWrite = writeBuf == appendBuf ? new byte[bufLength] : writeBuf;
        writeBuf = readBuf;
        readBuf = nextWrite;
    };

    rw.close();

            // now show the content of our file
    BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(f)));

    String line;
    while ((line = reader.readLine()) != null) {
        System.out.println(line);
    }
}
1 голос
/ 29 марта 2010

Нет. Здесь нет операций «внутрифайлового сдвига», только чтение и запись дискретных размеров.

0 голосов
/ 23 сентября 2017
private static void addPreAppnedText(File fileName) {
    FileOutputStream fileOutputStream =null;

    BufferedReader br = null;
    FileReader fr = null;
    String newFileName = fileName.getAbsolutePath() + "@";

    try {
        fileOutputStream = new FileOutputStream(newFileName);
        fileOutputStream.write("preappendTextDataHere".getBytes());

        fr = new FileReader(fileName);
        br = new BufferedReader(fr);

        String sCurrentLine;

        while ((sCurrentLine = br.readLine()) != null) {
            fileOutputStream.write(("\n"+sCurrentLine).getBytes());

        }
        fileOutputStream.flush();
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        try {
            fileOutputStream.close();
            if (br != null)
                br.close();

            if (fr != null)
                fr.close();

            new File(newFileName).renameTo(new File(newFileName.replace("@", "")));
        } catch (IOException ex) {
            ex.printStackTrace();
        }
    }
}
0 голосов
/ 29 марта 2010

Это можно было бы сделать, прочитав кусок файла, длина которого равна той, которую вы хотите добавить, записав новый контент вместо него, прочитав более поздний кусок и заменив его тем, что вы читали ранее, и и так далее, покачиваясь вниз до конца файла.

Однако не делайте этого, потому что если что-то останавливается (нехватка памяти, отключение питания, жульнический поток, вызывающий System.exit) в середине этого процесса, данные будут потеряны. Вместо этого используйте временный файл.

0 голосов
/ 29 марта 2010

Вы можете сохранить содержимое файла в String и добавить нужную строку, используя StringBuilder-Object. Вам просто нужно сначала поставить нужную строку, а затем добавить строку file-content-String.
Никаких дополнительных временных файлов не требуется.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...