23

我有一个进程会经常从 cron 调用,以读取其中包含某些与移动相关的命令的文件。我的进程需要读取和写入此数据文件 - 并保持锁定以防止其他进程在此期间接触它。用户可以执行一个完全独立的过程来(潜在的)写入/附加到这个相同的数据文件。我希望这两个进程运行良好,并且一次只能访问一个文件。

nio FileLock 似乎是我需要的(没有编写我自己的信号量类型文件),但我无法锁定它以供阅读。我可以锁定和写得很好,但是在阅读时尝试创建锁定时,我得到了一个 NonWritableChannelException。甚至可以锁定文件以供阅读吗?似乎 RandomAccessFile 更接近我的需要,但我不知道如何实现它。

这是失败的代码:

FileInputStream fin = new FileInputStream(f);
FileLock fl = fin.getChannel().tryLock();
if(fl != null) 
{
  System.out.println("Locked File");
  BufferedReader in = new BufferedReader(new InputStreamReader(fin));
  System.out.println(in.readLine());
          ...

在 FileLock 行上引发异常。

java.nio.channels.NonWritableChannelException
 at sun.nio.ch.FileChannelImpl.tryLock(Unknown Source)
 at java.nio.channels.FileChannel.tryLock(Unknown Source)
 at Mover.run(Mover.java:74)
 at java.lang.Thread.run(Unknown Source)

查看JavaDocs,它说

尝试写入最初未打开用于写入的通道时引发未经检查的异常。

但我不一定需要写信给它。当我尝试为写作目的创建 FileOutpuStream 等时,我很高兴直到我尝试在同一个文件上打开 FileInputStream。

4

3 回答 3

20
  1. 您是否知道锁定文件不会阻止其他进程触摸它,除非它们也使用锁?
  2. 您必须通过可写通道锁定。通过RandomAccessFile“rw”模式获取锁,然后打开您的FileInputStream. 确保两者都关闭!
于 2010-02-15T21:46:40.437 回答
13

如果您使用创建锁会更好tryLock(0L, Long.MAX_VALUE, true)

这会创建一个共享锁,这对于阅读来说是正确的。

tryLock()是 的简写tryLock(0L, Long.MAX_VALUE, false),即它请求独占写锁。

于 2010-05-29T12:38:02.540 回答
0

我写了一个测试程序和bash命令来确认文件锁的有效性:

import java.io.File;
import java.io.RandomAccessFile;
import java.nio.channels.FileLock;

public class FileWriterTest
{
    public static void main(String[] args) throws Exception
    {
        if (args.length != 4)
        {
            System.out.println("Usage: FileWriterTest <filename> <string> <sleep ms> <enable lock>");
            System.exit(1);
        }

        String filename = args[0];
        String data = args[1];
        int sleep = Integer.parseInt(args[2]);
        boolean enableLock = Boolean.parseBoolean(args[3]);

        try (RandomAccessFile raFile = new RandomAccessFile(new File(filename), "rw"))
        {
            FileLock lock = null;
            if (enableLock)
            {
                lock = raFile.getChannel().lock();
            }

            Thread.sleep(sleep);
            raFile.seek(raFile.length());
            System.out.println("writing " + data + " in a new line; current pointer = " + raFile.getFilePointer());
            raFile.write((data+"\n").getBytes());

            if (lock != null)
            {
                lock.release();
            }
        }
    }
}

使用此 bash 命令运行以检查它是否有效:

for i in {1..1000}
do
java FileWriterTest test.txt $i 10 true &
done

您应该看到每 10 毫秒(从输出)写入一次,最后所有数字都出现在文件中。

输出:

/tmp wc -l test.txt
1000 test.txt
/tmp

没有锁的相同测试显示数据丢失:

for i in {1..1000}
do
java FileWriterTest test.txt $i 10 false &
done

输出:

/tmp wc -l test.txt
764 test.txt
/tmp

修改它来测试 tryLock 应该很容易。

于 2019-11-15T06:30:41.097 回答