Java - 在读取时通过覆盖从文本文件中删除行

She*_*ley 8 java text-files printwriter

我试图从文本文件中删除一行文本而不复制到临时文件.我试图通过使用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();
 }
}
Run Code Online (Sandbox Code Playgroud)

它给了我奇怪的错误.它在堆栈跟踪中显示"NoSuchElementException"和"no line found".它指向不同的线条; 似乎任何nextLine()调用都可以做到这一点.是否可以通过这种方式删除一行?如果是这样,我做错了什么?如果没有,为什么?(顺便说一下,如果你想要这个,那么文本文件大概就是500行.不过,我不知道这是不是很重要,甚至不重要.)

aio*_*obe 21

正如其他人所指出的那样,如果你的程序在中途崩溃的风险最小,你可能最好使用临时文件:

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);

}
Run Code Online (Sandbox Code Playgroud)

(注意编码,新行字符和异常处理的草率处理.)


但是,我不喜欢回答" 我不会告诉你怎么做,因为你不应该这样做. ".(例如,在某些其他情况下,您可能正在处理一个大于硬盘驱动器一半的文件!)所以这里是:

你需要使用一个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();
}
Run Code Online (Sandbox Code Playgroud)