FileLock 多进程文件锁
FileLock是文件锁,它能保证同一时间只有一个进程(程序)能够修改它,或者都只可以读,这样就解决了多进程间的同步文件,保证了安全性。但是需要注意的是,它进程级别的,不是线程级别的,他可以解决多个进程并发访问同一个文件的问题,但是它不适用于控制同一个进程中多个线程对一个文件的访问。这也是为什么它叫做 多进程文件锁,而不是 多线程文件锁。
FileLock一般都是从FileChannel 中获取,FileChannel 提供了三个方法用以获取 FileLock。
- lock() 是阻塞式的,它要阻塞进程直到锁可以获得,或调用
lock()的线程中断,或调用lock()的通道关闭。 - tryLock()是非阻塞式的,它设法获取锁,但如果不能获得,例如因为其他一些进程已经持有相同的锁,而且不共享时,它将直接从方法调用返回。
position:锁定文件中的开始位置
size:锁定文件中的内容长度
shared:是否使用共享锁。true为共享锁;false为独占锁。
/**
* 先运行
* @throws Exception
*/
@Test
public void FileWriteTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
System.out.println("进程 1 开始写内容:" + LocalTime.now());
for (int i = 1; i <= 10; i++) {
randomAccessFile.writeUTF("VipSoft_" + i);
System.out.println("writeChars:" + i);
// 等待两秒
TimeUnit.SECONDS.sleep(2);
}
System.out.println("进程 1 完成写内容:" + LocalTime.now());
fileChannel.close();
randomAccessFile.close();
} /**
* 再运行,看读出文件大小的变化
* @throws Exception
*/
@Test
public void FileReadTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
System.out.println("开始读文件的时间:" + LocalTime.now()); for (int i = 0; i < 10; i++) {
// FileWriteTest() 运行后,运行 FileReadTest(),发现文件大小在变。说明文件在一边读一边写
System.out.println("文件大小为:" + randomAccessFile.length());
// 这里等待 1 秒
TimeUnit.SECONDS.sleep(1);
} System.out.println("结束读文件的时间:" + LocalTime.now());
fileChannel.close();
randomAccessFile.close();
}
使用文件锁,使用后,只能等锁释放掉以后,另一个进程才能对期进行操作
/**
* 先运行
* @throws Exception
*/
@Test
public void FileWriteTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock = fileChannel.lock();
System.out.println("进程 1 开始写内容:" + LocalTime.now());
for (int i = 1; i <= 10; i++) {
randomAccessFile.writeUTF("VipSoft_" + i);
System.out.println("writeChars:" + i);
// 等待两秒
TimeUnit.SECONDS.sleep(2);
}
System.out.println("进程 1 完成写内容:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
} /**
* 再运行,看读出文件大小的变化
* @throws Exception
*/
@Test
public void FileReadTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock = fileChannel.lock();
System.out.println("开始读文件的时间:" + LocalTime.now()); for (int i = 0; i < 10; i++) {
// FileWriteTest() 运行后,运行 FileReadTest(),发现文件大小在变。说明文件在一边读一边写
System.out.println("文件大小为:" + randomAccessFile.length());
// 这里等待 1 秒
TimeUnit.SECONDS.sleep(1);
} System.out.println("结束读文件的时间:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
}
同进程不同线程进行文件读写
FileLock是不适用同一进程不同线程之间文件的访问。因为你根本无法在一个进程中不同线程同时对一个文件进行加锁操作,如果线程1对文件进行了加锁操作,这时线程2也来进行加锁操作的话,则会直接抛出异常:java.nio.channels.OverlappingFileLockException
import org.junit.Test; import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.time.LocalTime;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; public class FileLockTest {
@Test
public void RunnableTest() throws InterruptedException {
Runner1 runner1 = new Runner1();
Runner2 runner2 = new Runner2();
Thread thread1 = new Thread(runner1);
Thread thread2 = new Thread(runner2);
thread1.start();
thread2.start();
System.out.print("阻塞当前线程,直到倒数计数器倒数到0");
new CountDownLatch(1).await();
} class Runner1 implements Runnable { // 实现了Runnable接口,jdk就知道这个类是一个线程
public void run() {
try {
FileWriteTest();
} catch (Exception e) {
e.printStackTrace();
}
}
} class Runner2 implements Runnable { // 实现了Runnable接口,jdk就知道这个类是一个线程
public void run() {
try {
FileReadTest();
} catch (Exception e) {
e.printStackTrace();
}
}
} /**
* 先运行
* @throws Exception
*/
public void FileWriteTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock = fileChannel.lock();
System.out.println("进程 1 开始写内容:" + LocalTime.now());
for (int i = 1; i <= 10; i++) {
randomAccessFile.writeUTF("VipSoft_" + i);
System.out.println("writeChars:" + i);
// 等待两秒
TimeUnit.SECONDS.sleep(2);
}
System.out.println("进程 1 完成写内容:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
} /**
* 再运行,看读出文件大小的变化
* @throws Exception
*/
@Test
public void FileReadTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock = fileChannel.lock();
System.out.println("开始读文件的时间:" + LocalTime.now()); for (int i = 0; i < 10; i++) {
// FileWriteTest() 运行后,运行 FileReadTest(),发现文件大小在变。说明文件在一边读一边写
System.out.println("文件大小为:" + randomAccessFile.length());
// 这里等待 1 秒
TimeUnit.SECONDS.sleep(1);
} System.out.println("结束读文件的时间:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
}
}
可以通过另外一种方式来规避程序异常,如下:
FileLock fileLock;
while (true){
try{
fileLock = fileChannel.tryLock();
break;
} catch (Exception e) {
System.out.println("这里是 FileWriteTest ,其他线程已经获取该文件锁了,当前线程休眠 2 秒再获取");
TimeUnit.SECONDS.sleep(2);
}
}
完整DEMO如下
package org.apache.rocketmq.store; import org.junit.Test; import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;
import java.nio.channels.FileLock;
import java.time.LocalTime;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; public class FileLockTest { @Test
public void RunnableTest() throws InterruptedException {
Runner1 runner1 = new Runner1();
Runner2 runner2 = new Runner2();
Thread thread1 = new Thread(runner1);
Thread thread2 = new Thread(runner2);
thread1.start();
thread2.start();
System.out.print("阻塞当前线程,直到倒数计数器倒数到0");
new CountDownLatch(1).await();
} class Runner1 implements Runnable {
public void run() {
try {
FileWriteTest();
} catch (Exception e) {
e.printStackTrace();
}
}
} class Runner2 implements Runnable {
public void run() {
try {
FileReadTest();
} catch (Exception e) {
e.printStackTrace();
}
}
} /**
* 先运行
* @throws Exception
*/
public void FileWriteTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock;
while (true){
try{
fileLock = fileChannel.tryLock();
break;
} catch (Exception e) {
System.out.println("这里是 FileWriteTest ,其他线程已经获取该文件锁了,当前线程休眠 2 秒再获取");
TimeUnit.SECONDS.sleep(2);
}
} System.out.println("进程 1 开始写内容:" + LocalTime.now());
for (int i = 1; i <= 10; i++) {
randomAccessFile.writeUTF("VipSoft_" + i);
System.out.println("writeChars:" + i);
// 等待两秒
TimeUnit.SECONDS.sleep(2);
}
System.out.println("进程 1 完成写内容:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
} /**
* 再运行,看读出文件大小的变化
* @throws Exception
*/
@Test
public void FileReadTest() throws Exception {
RandomAccessFile randomAccessFile = new RandomAccessFile("D:\\temp\\filelock.txt", "rw");
FileChannel fileChannel = randomAccessFile.getChannel();
// 这里是独占锁
FileLock fileLock;
while (true){
try{
fileLock = fileChannel.tryLock();
break;
} catch (Exception e) {
System.out.println("这里是 FileReadTest,其他线程已经获取该文件锁了,当前线程休眠 2 秒再获取");
TimeUnit.SECONDS.sleep(2);
}
}
System.out.println("开始读文件的时间:" + LocalTime.now()); for (int i = 0; i < 10; i++) {
// FileWriteTest() 运行后,运行 FileReadTest(),发现文件大小在变。说明文件在一边读一边写
System.out.println("文件大小为:" + randomAccessFile.length());
// 这里等待 1 秒
TimeUnit.SECONDS.sleep(1);
} System.out.println("结束读文件的时间:" + LocalTime.now());
// 完成后要释放掉锁
fileLock.release();
fileChannel.close();
randomAccessFile.close();
}
}
FileLock 多进程文件锁的更多相关文章
- Java NIO中的FileLock(文件锁)
FileLock,文件锁. 文件锁在OS中很常见,如果多个程序同时访问.修改同一个文件,很容易因为文件数据不同步而出现问题.给文件加一个锁,同一时间,只能有一个程序修改此文件,或者程序都只能读此文件, ...
- 【死磕NIO】— 跨进程文件锁:FileLock
大家好,我是大明哥,一个专注于[死磕 Java]系列创作的程序员. [死磕 Java ]系列为作者「chenssy」 倾情打造的 Java 系列文章,深入分析 Java 相关技术核心原理及源码 死磕 ...
- 文件锁FileLock
1.文件锁的定义 FileLock是文件锁,进程锁,用于进程间并发,控制不同程序(JVM)对同一文件的并发访问. FileLock是java 1.4 版本后出现的一个类,它可以通过对一个可写文件(w) ...
- Java进程间通信学习
转自:https://www.iteye.com/blog/polim-1278435 进程间通信的主要方法有:(1)管道(Pipe):管道可用于具有亲缘关系进程间的通信,允许一个进程和另一个与它有共 ...
- 《JAVA程序设计》_第六周学习总结
一.本周学习内容 1.内部类--7.1知识 在一个类的内部定义的类成为内部类,包含内部类的类叫做外嵌类 内部类和外嵌类的关系 外嵌类的成员变量在内部类中仍然有效,内部类也可调用外嵌类中的方法 内部类的 ...
- 20175212童皓桢 《Java程序设计》第六周学习总结
20175212童皓桢 <Java程序设计>第六周学习总结 教材学习内容总结 第七章 内部类与异常类 1.内部类 Java支持在一个类中定义另一个类,这样的类称作内部类,包含内部类的类称为 ...
- 海纳百川而来的一篇相当全面的Java NIO教程
目录 零.NIO包 一.Java NIO Channel通道 Channel的实现(Channel Implementations) Channel的基础示例(Basic Channel Exampl ...
- 20165237 2017-2018-2 《Java程序设计》第5周学习总结
20165237 2017-2018-2 <Java程序设计>第5周学习总结 教材学习内容总结 1.内部类:在一个类中定义另一个类:外嵌类:包含内部类的类. 2.内部类的类体中不能声明类变 ...
- 20175314 《Java程序设计》第六周学习总结
20175314 <Java程序设计>第六周学习总结 教材学习内容总结 第七章:内部类与异常类 内部类:内部类就是在一个类中再定义一个类,这个在类中定义的类就叫做内部类,而包含内部类的类叫 ...
- java程序怎么在一个电脑上只启动一次,只开一个进程
目录 <linux文件锁flock> <NIO文件锁FileLock> <java程序怎么在一个电脑上只启动一次,只开一个进程> 方案1: 单进程程序可以用端口绑定 ...
随机推荐
- NET8 ORM 使用AOT SqlSugar
AOT介绍 .Net8的本地预编机器码AOT,它几乎进行了100%的自举.微软为了摆脱C++的钳制,做了很多努力.也就是代码几乎是用C#重写,包括了虚拟机,GC,内存模型等等.而需要C++做的,也就仅 ...
- 放弃"Jenkins"的种种理由,期待更好赋能研发的持续交付平台
Jenkins 很酷,但是不完美,有历史局限性造成的问题.本文仅从"如何更好给研发团队赋能的角度",剖析Jenkins, 探讨理想的持续交付平台, 不带货无广告- 不完美的Jenk ...
- CART算法解密:从原理到Python实现
本文深入探讨了CART(分类与回归树)算法的核心原理.实现方法以及应用场景.文章首先介绍了决策树的基础知识,然后详细解析了CART算法的工作机制,包括特征选择和树的构建.接着,通过Python和PyT ...
- Linux笔记03: Linux常用命令_3.2目录操作命令
3.2 目录操作命令 3.2.1 ls命令 ●命令名称:ls. ●英文原意:list directory contents. ●所在路径:/usr/bin/ls. ●执行权限:所有用户. ●功能描述: ...
- python数据类型元组、列表、集合、字典相互嵌套
系统 Windows 10 专业工作站版22H2 软件 python-3.9.6-amd64.exe 拓展库: jupyter==1.0.0 notebook==7.0.6 1.元组嵌套 1.1 元组 ...
- HDFS存储原理
冗余数据保存问题: 一个数据块默认被保存三次 好处:1.加快数据传输错误(假如要同时访问数据块1 因为他冗余存储就会有3份 所以会加快数据传输速度) 2.很容易检查数据错误 3.保证数据可靠性 数据的 ...
- GZY.Quartz.MUI(基于Quartz的UI可视化操作组件) 2.6.0发布 兼容.Net8.0
前言 为了迎接.Net8.0 2.6.0终于发布了~ 更新内容: 兼容.NET8.0 新增界面按分组名称排序功能 优化本地持久化时文件路径异常的问题 优化数据库持久化时偶现的异常问题 新增简易授权,增 ...
- 重磅:谷歌发布最强大AI模型【Google Gemini】
一.前言 北京时间 2023年12 月 13 日Google 发布了最新的 Gemini Pro模型,并且提供了 API 访问. 一个更好的消息是:Gemini Pro 可免费使用.赶紧体验起来吧~ ...
- 牛客小白月赛2 F题黑黑白白 (博弈或dfs)
题目链接:https://www.nowcoder.com/acm/contest/86/F 解题思路:赛后看博客都说是sg函数.emmm,后面看了别人代码dfs也可以,只要找到一条能赢的路就可以. ...
- 数字孪生与GIS结合趋势背后,是市场需求的变化
随着数字化时代的来临,数字孪生和地理信息系统(GIS)作为两个独立的技术领域,正日益融合并发挥着协同作用.这一趋势的背后,是市场需求的变化和对更智能.更精准.更实用的解决方案的追求. 数字孪生与GIS ...