Java NIO FileLock

FileLock简介

文件锁在OS中很常见,如果多个程序同时访问、修改一个文件,很容易因为文件数据不同步而出现问题。给文件加一个锁,同一时间,只能有一个程序修改此文件,或者程序都只能读此文件,这就解决了同步问题。

文件锁是进程级别的,不是线程级别的。文件锁可以解决多个进程并发访问、修改同一文件的问题,但不能解决多线程并发访问、修改同一文件的问题。

使用文件锁时,同一进程内的多个线程,可以同时访问、修改此文件。

文件锁是当前程序所属的JVM实例持有的,一旦获取到文件锁,要调用release(),或者关闭对应的FileChannel对象,或者当前JVM退出,才会释放这个锁。

一旦某个进程(比如JVM实例)对某个文件加锁,则在释放这个锁之前,此进程不能再对此文件加锁,就是说JVM实例在同一文件上的文件锁是不重叠的(进程级别不能重复在同一文件上获取锁)。

文件锁分类

  • 排它锁:又叫独占锁。对文件加排它锁后,该进程可以对此文件进行读写,该进程独占此文件,其他进程不能读写此文件,直到该进程释放文件锁。
  • 共享锁:某个进程对文件加共享锁,其他进程也可以访问此文件,但这些进程都只能读此文件,不能写。线程是安全的。只要还有一个进程持有共享锁,此文件只能读,不能写。

获取文件锁方法

  • lock():对整个文件加锁,默认排它锁。
  • lock(long position, long size, boolean shared):自定义加锁方式。前2个参数指定要加锁的部分(可只对此文件的部分内容加锁),第三个参数值指定是否是共享锁。
  • tryLock():对整个文件加锁,默认排它锁。
  • trylock(long position, long size, boolean shared):自定义加锁方式。如果指定为共享锁,则其他进程可读此文件,所有进程均不能写此文件,如果某进程试图对此文件进行写操作,会抛出异常。

lock与tryLock的区别

lock是阻塞式的,如果未获取到文件锁,会一直阻塞当前线程,直到获取文件锁。tryLock和lock的作用相同,只不过tryLock是非阻塞的,tryLock是尝试获取文件锁,获取成功就返回锁对象,否则返回null,不会阻塞当前线程。

FileLock两个方法

  • boolean isShared() :此文件锁是否是共享锁。
  • boolean isValid():此文件锁是否还有效。

使用示例

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;

public class FileLockDemo1 {

    public static void main(String[] args) throws IOException {
        String input = "abc";
        System.out.println(input);

        ByteBuffer buf = ByteBuffer.wrap(input.getBytes());

        String filePath = "d:\\qrxqrx\\01.txt";
        Path path = Paths.get(filePath);
        FileChannel fileChannel = FileChannel.open(path, StandardOpenOption.WRITE,StandardOpenOption.APPEND);
        fileChannel.position(fileChannel.size()-1);

        //写文件
        FileLock lock = fileChannel.lock();
        System.out.println("是否为共享锁:"+lock.isShared());
        fileChannel.write(buf);
        fileChannel.close();

        //读文件
        FileReader fileReader = new FileReader(filePath);
        BufferedReader bufferedReader = new BufferedReader(fileReader);
        String s = bufferedReader.readLine();
        System.out.println(s);
        fileReader.close();
        bufferedReader.close();
    }

}

上一篇:代码JAR库总是只一个


下一篇:10天拿到腾讯Java岗offer,阿里Java面试流程