JAVA文件锁

项目使用到集群环境,流程发送时如果确保一个流程不会被两个流程同时调用?

有一种办法是用文件锁的方式来实现。

代码如下:

锁接口:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.FileNotFoundException;
  3. importjava.io.IOException;
  4. publicinterfaceLock{
  5. /**
  6. *检测是否被锁定
  7. *@returntrue被锁定,false空闲
  8. **/
  9. publicabstractbooleanisLocked()throwsFileNotFoundException;
  10. /**
  11. *获取锁资源
  12. *@returntrue成功锁定目标资源,false锁定操作失败
  13. **/
  14. publicabstractbooleanobtain()throwsIOException;
  15. /**
  16. *释放锁
  17. **/
  18. publicabstractvoidunlock();
  19. }

文件锁的实现:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.File;
  3. importjava.io.FileNotFoundException;
  4. importjava.io.IOException;
  5. importjava.io.RandomAccessFile;
  6. importjava.nio.channels.FileChannel;
  7. importjava.nio.channels.FileLock;
  8. importjava.nio.channels.OverlappingFileLockException;
  9. publicclassFileProgrameLockimplementsLock{
  10. privateStringcallerThreadID=null;
  11. privateStringlockFileName=null;
  12. FileChannelchannel=null;
  13. privateFileLocklock=null;
  14. publicstaticLockget(StringfileName,StringcallerThreadID){
  15. FileProgrameLockd=newFileProgrameLock(fileName);
  16. d.callerThreadID=callerThreadID;
  17. return(Lock)d;
  18. }
  19. publicFileProgrameLock(StringlockFileName){
  20. this.lockFileName=lockFileName;
  21. }
  22. /**
  23. *检测是否被锁定-不建议使用
  24. *@returntrue被锁定,false空闲
  25. *@deprecated
  26. **/
  27. publicbooleanisLocked()throwsFileNotFoundException{
  28. Filetf=newFile(lockFileName);
  29. if(!tf.exists()){
  30. returnfalse;
  31. }
  32. FileChannel__channel=newRandomAccessFile(tf,"rw").getChannel();
  33. FileLocktl=null;
  34. try{
  35. tl=__channel.tryLock();
  36. if(tl==null){
  37. returntrue;
  38. }else{
  39. returnfalse;
  40. }
  41. }catch(OverlappingFileLockExceptione){
  42. returntrue;
  43. }catch(IOExceptione){
  44. returntrue;
  45. }catch(Exceptione){
  46. returntrue;
  47. }finally{
  48. try{
  49. if(tl!=null){
  50. tl.release();
  51. }
  52. tl=null;
  53. if(__channel.isOpen()){
  54. __channel.close();
  55. }
  56. __channel=null;
  57. tf=null;
  58. }catch(IOExceptione){
  59. e.printStackTrace();
  60. }
  61. }
  62. }
  63. /**
  64. *获取锁资源
  65. *@returntrue成功锁定目标资源,false锁定操作失败
  66. **/
  67. publicbooleanobtain()throwsIOException{
  68. Filetf=newFile(lockFileName);
  69. createFile();
  70. channel=newRandomAccessFile(tf,"rw").getChannel();
  71. try{
  72. //System.out.println("getlock000>>>>>>>>>>>>>>>");
  73. lock=channel.lock();
  74. //System.out.println("getlock>>>>>>>>>>>>>>>");
  75. returntrue;
  76. }catch(OverlappingFileLockExceptione){
  77. returnfalse;
  78. }catch(Exceptione){
  79. returnfalse;
  80. }
  81. }
  82. /**
  83. *释放锁
  84. **/
  85. publicvoidunlock(){
  86. try{
  87. if(lock!=null){
  88. lock.release();
  89. }
  90. System.out.println(callerThreadID+"unlockXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX");
  91. if(channel!=null&&channel.isOpen()){
  92. channel.close();
  93. }
  94. lock=null;
  95. channel=null;
  96. this.deleteFile();
  97. }catch(IOExceptione){
  98. }
  99. }
  100. protectedvoidfinalize()throwsThrowable{
  101. System.out.println(callerThreadID+this.getClass()+".finalize()");
  102. super.finalize();
  103. }
  104. privatevoidcreateFile()throwsIOException{
  105. try{
  106. Filetf=newFile(lockFileName);
  107. if(!tf.exists()){
  108. tf.createNewFile();
  109. }
  110. tf=null;
  111. }catch(IOExceptione){
  112. System.out.println(e+lockFileName);
  113. throwe;
  114. }
  115. }
  116. privatevoiddeleteFile(){
  117. Filetf=newFile(lockFileName);
  118. if(tf.exists()){
  119. tf.delete();
  120. }
  121. tf=null;
  122. }
  123. }

工厂类:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.File;
  3. importjava.io.FileNotFoundException;
  4. importjava.io.IOException;
  5. publicclassMakeLockimplementsRunnable{
  6. privateStringthreadID="";
  7. publicvoidrun(){
  8. try{
  9. while(true){
  10. test2(threadID);
  11. Thread.sleep(200);
  12. }
  13. }catch(IOExceptione){
  14. System.out.println(e);
  15. e.printStackTrace();
  16. }catch(InterruptedExceptione){
  17. System.out.println(e);
  18. }
  19. }
  20. publicvoidtest2(StringthreadID)throwsFileNotFoundException,IOException,InterruptedException{
  21. Locklock=newMakeLock(threadID).getLock("c:/001/lockfile001.lock",threadID);
  22. System.out.println(threadID+":obtain...");
  23. booleanb=lock.obtain();
  24. //当有重叠时会发生等待,所以外侧先执行isLocked()判断
  25. System.out.println(threadID+":obtained"+b);
  26. if(b){//执行业务逻辑
  27. Thread.sleep(390);
  28. for(inti=0;i<Integer.MAX_VALUE;i++){
  29. ;
  30. }
  31. lock.unlock();
  32. }
  33. lock=null;
  34. }
  35. publicMakeLock(StringthreadID){
  36. this.threadID=threadID;
  37. }
  38. publicLockgetLock(Stringname,StringthreadID){
  39. finalStringBufferbuf=newStringBuffer();
  40. returnFileProgrameLock.get(name,threadID);
  41. }
  42. }

使用方法:

Java代码 收藏代码
  1. publicvoidtest2(StringthreadID)throwsFileNotFoundException,IOException,InterruptedException{
  2. Locklock=newMakeLock(threadID).getLock("c:/001/lockfile001.lock",threadID);
  3. if(!lock.isLocked()){
  4. System.out.println(threadID+":obtain...");
  5. booleanb=lock.obtain();
  6. System.out.println(threadID+":obtained"+b);
  7. if(b){//执行业务逻辑
  8. Thread.sleep(390);
  9. for(inti=0;i<Integer.MAX_VALUE;i++){
  10. ;
  11. }
  12. lock.unlock();
  13. }
  14. }else{
  15. System.out.println(threadID+":can'tgetalock:"+lock);
  16. }
  17. lock=null;
  18. }

多线程调用测试:

Java代码 收藏代码
  1. publicstaticvoidmain(String[]args){
  2. //newAAA().test();
  3. System.out.println("=========================================");
  4. Threadth1=newThread(newMakeLock("====================thread1===================="));
  5. Threadth2=newThread(newMakeLock("####################thread2####################"));
  6. Threadth3=newThread(newMakeLock("@@@@@@@@@@@@@@@@@@@@thread3@@@@@@@@@@@@@@@@@@@@"));
  7. Threadth4=newThread(newMakeLock("$$$$$$$$$$$$$$$$$$$$thread4$$$$$$$$$$$$$$$$$$$$"));
  8. Threadth5=newThread(newMakeLock("&&&&&&&&&&&&&&&&&&&&thread5&&&&&&&&&&&&&&&&&&&&"));
  9. th1.start();
  10. th2.start();
  11. th3.start();
  12. th4.start();
  13. th5.start();
  14. }

=================

经测试可以在共享文件系统下工作。

附件中AAA.java多线程测试代码可以直接运行。

项目使用到集群环境,流程发送时如果确保一个流程不会被两个流程同时调用?

有一种办法是用文件锁的方式来实现。

代码如下:

锁接口:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.FileNotFoundException;
  3. importjava.io.IOException;
  4. publicinterfaceLock{
  5. /**
  6. *检测是否被锁定
  7. *@returntrue被锁定,false空闲
  8. **/
  9. publicabstractbooleanisLocked()throwsFileNotFoundException;
  10. /**
  11. *获取锁资源
  12. *@returntrue成功锁定目标资源,false锁定操作失败
  13. **/
  14. publicabstractbooleanobtain()throwsIOException;
  15. /**
  16. *释放锁
  17. **/
  18. publicabstractvoidunlock();
  19. }

文件锁的实现:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.File;
  3. importjava.io.FileNotFoundException;
  4. importjava.io.IOException;
  5. importjava.io.RandomAccessFile;
  6. importjava.nio.channels.FileChannel;
  7. importjava.nio.channels.FileLock;
  8. importjava.nio.channels.OverlappingFileLockException;
  9. publicclassFileProgrameLockimplementsLock{
  10. privateStringcallerThreadID=null;
  11. privateStringlockFileName=null;
  12. FileChannelchannel=null;
  13. privateFileLocklock=null;
  14. publicstaticLockget(StringfileName,StringcallerThreadID){
  15. FileProgrameLockd=newFileProgrameLock(fileName);
  16. d.callerThreadID=callerThreadID;
  17. return(Lock)d;
  18. }
  19. publicFileProgrameLock(StringlockFileName){
  20. this.lockFileName=lockFileName;
  21. }
  22. /**
  23. *检测是否被锁定-不建议使用
  24. *@returntrue被锁定,false空闲
  25. *@deprecated
  26. **/
  27. publicbooleanisLocked()throwsFileNotFoundException{
  28. Filetf=newFile(lockFileName);
  29. if(!tf.exists()){
  30. returnfalse;
  31. }
  32. FileChannel__channel=newRandomAccessFile(tf,"rw").getChannel();
  33. FileLocktl=null;
  34. try{
  35. tl=__channel.tryLock();
  36. if(tl==null){
  37. returntrue;
  38. }else{
  39. returnfalse;
  40. }
  41. }catch(OverlappingFileLockExceptione){
  42. returntrue;
  43. }catch(IOExceptione){
  44. returntrue;
  45. }catch(Exceptione){
  46. returntrue;
  47. }finally{
  48. try{
  49. if(tl!=null){
  50. tl.release();
  51. }
  52. tl=null;
  53. if(__channel.isOpen()){
  54. __channel.close();
  55. }
  56. __channel=null;
  57. tf=null;
  58. }catch(IOExceptione){
  59. e.printStackTrace();
  60. }
  61. }
  62. }
  63. /**
  64. *获取锁资源
  65. *@returntrue成功锁定目标资源,false锁定操作失败
  66. **/
  67. publicbooleanobtain()throwsIOException{
  68. Filetf=newFile(lockFileName);
  69. createFile();
  70. channel=newRandomAccessFile(tf,"rw").getChannel();
  71. try{
  72. //System.out.println("getlock000>>>>>>>>>>>>>>>");
  73. lock=channel.lock();
  74. //System.out.println("getlock>>>>>>>>>>>>>>>");
  75. returntrue;
  76. }catch(OverlappingFileLockExceptione){
  77. returnfalse;
  78. }catch(Exceptione){
  79. returnfalse;
  80. }
  81. }
  82. /**
  83. *释放锁
  84. **/
  85. publicvoidunlock(){
  86. try{
  87. if(lock!=null){
  88. lock.release();
  89. }
  90. System.out.println(callerThreadID+"unlockXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX");
  91. if(channel!=null&&channel.isOpen()){
  92. channel.close();
  93. }
  94. lock=null;
  95. channel=null;
  96. this.deleteFile();
  97. }catch(IOExceptione){
  98. }
  99. }
  100. protectedvoidfinalize()throwsThrowable{
  101. System.out.println(callerThreadID+this.getClass()+".finalize()");
  102. super.finalize();
  103. }
  104. privatevoidcreateFile()throwsIOException{
  105. try{
  106. Filetf=newFile(lockFileName);
  107. if(!tf.exists()){
  108. tf.createNewFile();
  109. }
  110. tf=null;
  111. }catch(IOExceptione){
  112. System.out.println(e+lockFileName);
  113. throwe;
  114. }
  115. }
  116. privatevoiddeleteFile(){
  117. Filetf=newFile(lockFileName);
  118. if(tf.exists()){
  119. tf.delete();
  120. }
  121. tf=null;
  122. }
  123. }

工厂类:

Java代码 收藏代码
  1. packagelock;
  2. importjava.io.File;
  3. importjava.io.FileNotFoundException;
  4. importjava.io.IOException;
  5. publicclassMakeLockimplementsRunnable{
  6. privateStringthreadID="";
  7. publicvoidrun(){
  8. try{
  9. while(true){
  10. test2(threadID);
  11. Thread.sleep(200);
  12. }
  13. }catch(IOExceptione){
  14. System.out.println(e);
  15. e.printStackTrace();
  16. }catch(InterruptedExceptione){
  17. System.out.println(e);
  18. }
  19. }
  20. publicvoidtest2(StringthreadID)throwsFileNotFoundException,IOException,InterruptedException{
  21. Locklock=newMakeLock(threadID).getLock("c:/001/lockfile001.lock",threadID);
  22. System.out.println(threadID+":obtain...");
  23. booleanb=lock.obtain();
  24. //当有重叠时会发生等待,所以外侧先执行isLocked()判断
  25. System.out.println(threadID+":obtained"+b);
  26. if(b){//执行业务逻辑
  27. Thread.sleep(390);
  28. for(inti=0;i<Integer.MAX_VALUE;i++){
  29. ;
  30. }
  31. lock.unlock();
  32. }
  33. lock=null;
  34. }
  35. publicMakeLock(StringthreadID){
  36. this.threadID=threadID;
  37. }
  38. publicLockgetLock(Stringname,StringthreadID){
  39. finalStringBufferbuf=newStringBuffer();
  40. returnFileProgrameLock.get(name,threadID);
  41. }
  42. }

使用方法:

Java代码 收藏代码
  1. publicvoidtest2(StringthreadID)throwsFileNotFoundException,IOException,InterruptedException{
  2. Locklock=newMakeLock(threadID).getLock("c:/001/lockfile001.lock",threadID);
  3. if(!lock.isLocked()){
  4. System.out.println(threadID+":obtain...");
  5. booleanb=lock.obtain();
  6. System.out.println(threadID+":obtained"+b);
  7. if(b){//执行业务逻辑
  8. Thread.sleep(390);
  9. for(inti=0;i<Integer.MAX_VALUE;i++){
  10. ;
  11. }
  12. lock.unlock();
  13. }
  14. }else{
  15. System.out.println(threadID+":can'tgetalock:"+lock);
  16. }
  17. lock=null;
  18. }

多线程调用测试:

Java代码 收藏代码
  1. publicstaticvoidmain(String[]args){
  2. //newAAA().test();
  3. System.out.println("=========================================");
  4. Threadth1=newThread(newMakeLock("====================thread1===================="));
  5. Threadth2=newThread(newMakeLock("####################thread2####################"));
  6. Threadth3=newThread(newMakeLock("@@@@@@@@@@@@@@@@@@@@thread3@@@@@@@@@@@@@@@@@@@@"));
  7. Threadth4=newThread(newMakeLock("$$$$$$$$$$$$$$$$$$$$thread4$$$$$$$$$$$$$$$$$$$$"));
  8. Threadth5=newThread(newMakeLock("&&&&&&&&&&&&&&&&&&&&thread5&&&&&&&&&&&&&&&&&&&&"));
  9. th1.start();
  10. th2.start();
  11. th3.start();
  12. th4.start();
  13. th5.start();
  14. }

=================

经测试可以在共享文件系统下工作。

附件中AAA.java多线程测试代码可以直接运行。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值