我正在尝试从文本文件中删除一行文本而不复制到临时文件。我试图通过使用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行。不过,我不知道这个数字有多大甚至很重要。)
答案 0 :(得分:20)
正如其他人所指出的那样,如果您的程序在中途崩溃的风险最小,那么最好使用临时文件:
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);
}
(谨防编码,新行字符和异常处理的草率处理。)
但是,我不喜欢回答问题“我不会告诉你怎么做,因为你不应该这样做。”。 (例如,在某些其他情况下,您可能正在处理一个大于硬盘驱动器一半的文件!)所以这里有:
您需要使用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();
}
答案 1 :(得分:2)
你不能这样做。 FileWriter只能追加到一个文件,而不是写在它的中间 - 如果你想在中间写,你需要RandomAccessFile。你现在做什么 - 你第一次写文件时覆盖文件(它变空了 - 这就是你得到异常的原因)。您可以使用append标志设置为true来创建FileWriter - 但是这样您可以附加到文件而不是在其中间写入。
我真的建议写一个新文件,然后在最后重命名。
答案 2 :(得分:1)