java - 线程和进程之间的文件锁

标签 java multithreading file-io file-locking

我正在编写一个程序,该程序从同一 JVM 上的不同线程和不同的 JVM/进程写入单个文件。有没有一种方法可以为线程和进程锁定一个文件,这样无论有多少线程/进程同时尝试写入,一次只能写入一个文件?

目前我有类似于以下的东西,它适用于锁定线程,但不适用于阻塞进程。如果我尝试在下面的实现之上使用 FileLock,它似乎 synchronized 停止工作。

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import scripts.com.adz.commons.utils.FileUtilities;

import java.io.*;

public class Foo {
    public static void main(String[] args) throws IOException {
        Bar bar = new Bar();
        bar.start();

        while (true) {
            FileUtilities.writeObjectToFile("C:\\test.html", "foo");
        }
    }
}

class Bar extends Thread {
    @Override
    public void run() {
        while (true) {
            try {
                FileUtilities.writeObjectToFile("C:\\test.html", "bar");
            } catch (IOException ignored) {}
        }
    }
}

class FileUtilitiess {
    private static final Object _lock = new Object();

    public static <T> T readObjectFromFile(File file) throws IOException, ClassNotFoundException {
        synchronized (_lock) {
            final byte[] bytes = FileUtils.readFileToByteArray(file);

            ByteArrayInputStream bis = null;
            ObjectInputStream ois = null;

            try {
                ois = new ObjectInputStream(bis = new ByteArrayInputStream(bytes));

                return (T) ois.readObject();
            } finally {
                IOUtils.closeQuietly(ois);
                IOUtils.closeQuietly(bis);
            }
        }
    }

    public static void writeObjectToFile(File file, Object object) throws IOException {
        System.out.println("Sent object: " + object.toString());
        synchronized (_lock) {
            System.out.println("Writing object: " + object.toString());

            ByteArrayOutputStream bos = null;
            ObjectOutputStream oos = null;

            try {
                oos = new ObjectOutputStream(bos = new ByteArrayOutputStream());
                oos.writeObject(object);

                FileUtils.writeByteArrayToFile(file, bos.toByteArray());

                // - Start: For testing lock.
                try {
                    Thread.sleep(10000);
                } catch (InterruptedException ignored) {}
                // - End: For testing lock.
            } finally {
                IOUtils.closeQuietly(oos);
                IOUtils.closeQuietly(bos);
            }
        }
    }
}

最佳答案

参见 FileLock javadoc :

File locks are held on behalf of the entire Java virtual machine.

这意味着在操作系统级别上,您的应用程序的不同线程将具有相同的访问锁定区域的权限。

要锁定来自不同线程的文件访问,您必须封装文件 IO 代码并强制执行同步。

关于java - 线程和进程之间的文件锁,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28113009/

相关文章:

java - Spring @asynch 线程并发共享资源

java - Spring Config 属性未在某个服务器上填充

vb.net - vb.net-后台线程问题

python - 将列表项转换为 int 并将它们相加的最有效方法

java - 将 JSONObject 写入文件

javascript数组到数组

java - Tomcat ClassNotFoundException : org. slf4j.spi.LoggerFactoryBinder 但 slf4j 在库中

java - 如何在java中使用jdom从节点中删除子节点?

C++11 线程不适用于虚拟成员函数

c++ - QAbstractItemModel 子类线程安全吗?