Java – 在读取文本文件时通过覆盖删除文本行

我试图从文本文件中删除一行文本而不复制到临时文件。 我试图通过使用Printwriter和扫描仪,并让他们在同一时间遍历文件,作家写什么扫描仪读取和覆盖每一行相同的东西,直到它到达我想要的行删除。 然后,我推进扫描仪,但不是作家,并继续像以前一样。 这里是代码:

但首先,参数:我的文件名是数字,所以这将读取1.txt或2.txt等,所以f指定的文件名。 我将它转换为一个文件的构造函数中的string。 Int n是我想要删除的行的索引。

public void deleteLine(int f, int n){ try{ Scanner reader = new Scanner(new File(f+".txt")); PrintWriter writer = new PrintWriter(new FileWriter(new File(f+".txt")),false); for(int w=0; w<n; w++) writer.write(reader.nextLine()); reader.nextLine(); while(reader.hasNextLine()) writer.write(reader.nextLine()); } catch(Exception e){ System.err.println("Enjoy the stack trace!"); e.printStackTrace(); } } 

它给了我奇怪的错误。 它在堆栈跟踪中显示“NoSuchElementException”和“no line found”。 它指向不同的路线; nextLine()调用似乎可以做到这一点。 是否可以删除一条线? 如果是这样,我做错了什么? 如果没有,为什么? (顺便说一句,如果你想要的话,这个文本文件大概是500行,不过我不知道这个数字是不是很大,甚至是很重要)。

正如其他人所指出的那样,如果程序崩溃的风险很小,那么使用临时文件可能会更好:

 public static void removeNthLine(String f, int toRemove) throws IOException { File tmp = File.createTempFile("tmp", ""); BufferedReader br = new BufferedReader(new FileReader(f)); BufferedWriter bw = new BufferedWriter(new FileWriter(tmp)); for (int i = 0; i < toRemove; i++) bw.write(String.format("%s%n", br.readLine())); br.readLine(); String l; while (null != (l = br.readLine())) bw.write(String.format("%s%n", l)); br.close(); bw.close(); File oldFile = new File(f); if (oldFile.delete()) tmp.renameTo(oldFile); } 

(注意对编码,新行字符和exception处理的粗糙处理。)


然而,我不喜欢回答问题,“ 我不会告诉你怎么做,因为你不应该这样做 ”。 (例如,在其他一些情况下,您可能正在使用比硬盘大一半的文件!)所以在这里:

你需要使用一个RandomAccessFile 。 使用这个类,你可以使用同一个对象来读写文件:

 public static void removeNthLine(String f, int toRemove) throws IOException { RandomAccessFile raf = new RandomAccessFile(f, "rw"); // Leave the n first lines unchanged. for (int i = 0; i < toRemove; i++) raf.readLine(); // Shift remaining lines upwards. long writePos = raf.getFilePointer(); raf.readLine(); long readPos = raf.getFilePointer(); byte[] buf = new byte[1024]; int n; while (-1 != (n = raf.read(buf))) { raf.seek(writePos); raf.write(buf, 0, n); readPos += n; writePos += n; raf.seek(readPos); } raf.setLength(writePos); raf.close(); } 

你不能这样做。 FileWriter只能附加到一个文件,而不是写在它的中间 – 如果你想写在中间,你需要RandomAccessFile。 你现在所做的 – 你第一次写入文件时覆盖了文件(它变成空的 – 这就是为什么你会得到这个exception)。 您可以创buildFileWriter,并将append标志设置为true – 但这样您可以追加到文件而不是写在文件的中间。

我真的build议写一个新的文件,然后在最后重命名。

@shelley:你不能做你想做的事情,而且你不应该做。 你应该阅读这个文件并写入一个临时文件,原因有几个,有可能这样做(而不是你正在做的),另一个是如果进程被破坏,你可能会打包而不会丢失原始文件。 现在,您可以使用RandomAccessFile更新文件的特定位置,但通常是在处理固定大小的logging而非典型的文本文件时完成的(以我的经验)。