去故就新 Java线程新同步机制

开发 后端
新的Java线程同步机制,主要对可重入锁ReentrantLock,条件对象Condition以及读写锁ReentrantReadWriteLock机制解析。

1、可重入锁ReentrantLock,相当于synchronized块,为临界区提供互斥访问机制。

(1) 相关的接口

创建一个可重入锁

  1. Lock lock = new ReentrantLock(); 

请求锁,如果锁被当前另一个线程持有,则阻塞。

  1. void lock();

释放锁

  1. void unlock(); 

非阻塞型lock()

  1. boolean tryLock(); 


(2) 使用基本结构

  1. locker.lock(); 
  2. try
  3. //code here to access the cirtical section 
  4. }finally
  5. locker.unlock(); 

这种结构保证在任何时刻只有一个线程能够进入临界区,如果一个线程锁住了锁对象,其他任何线程在调用lock时,都会被阻塞,直到第一个线程释放锁对象。而且无论try块是否抛出异常,都会执行finally block,解锁locker。

(3) 锁的可重入性

锁是可重入的,线程能够重复地获取它已经拥有的锁。锁对象维护一个持有计数(hold count)来追踪对lock方法的嵌套调用。线程在每次调用lock后都要调用unlock来释放锁。由于这个特性,被一个锁保护的代码可以调用另一个使用相同锁的方法。

(4) 示例代码:

  1. import java.util.concurrent.locks.Lock; 
  2. import java.util.concurrent.locks.ReentrantLock; 
  3.  
  4. class WorkerOne extends Thread{ 
  5.     private Lock locker; 
  6.     public WorkerOne (Lock locker){ 
  7.         this.locker = locker; 
  8.     } 
  9.      
  10.     public void run(){ 
  11.         locker.lock(); 
  12.         try
  13. System.out.println(Thread.currentThread().getName()+":step into critical section"); 
  14.         }finally
  15.             locker.unlock();     
  16.         } 
  17.     } 
  18.  
  19. class WorkerTwo extends Thread{ 
  20.     private Lock locker; 
  21.     public WorkerTwo (Lock locker){ 
  22.         this.locker = locker; 
  23.     } 
  24.      
  25.     public void sayHello(){ 
  26.         locker.lock(); 
  27.         try{    System.out.println(Thread.currentThread().getName()+":call sayHello()"); 
  28.             Thread.sleep(1000); 
  29.         } catch (InterruptedException e) { 
  30.             e.printStackTrace(); 
  31.         }finally
  32.             locker.unlock(); 
  33.         } 
  34.     } 
  35.      
  36.     public void run(){ 
  37.         locker.lock();   
  38.         try{        System.out.println(Thread.currentThread().getName()+":setp into critical section"); 
  39.                         //测试锁的可重入性  
  40.             sayHello(); 
  41.         }finally
  42.             locker.unlock();     
  43.         } 
  44.     } 
  45.  
  46. public class Test5 { 
  47.     public static void main(String[] args) { 
  48.         Lock locker = new ReentrantLock(); 
  49.         WorkerOne wo= new WorkerOne(locker); 
  50.         wo.setName("WorkerOne"); 
  51.         WorkerTwo wt = new WorkerTwo(locker); 
  52.         wt.setName("WorkerTwo"); 
  53.          
  54.         wt.start(); 
  55.         wo.start();  
  56.     } 

输出:

WorkerTwo:setp into critical section
WorkerTwo:call sayHello()
WorkerOne:step into critical section

2、条件对象Condition,相当于wait-notify机制,提供一种线程间的等待通知机制,condition中的等待-通知方法是await(),signal(),signalAll(),也需要在互斥环境下被调用。

(1) 相关的接口

创建Condition对象,Condition对象是跟Lock关联在一起的。

  1. Lock locker = new ReentrantLock(); 
  2. Condition cond = locker.newCondition(); 

把此线程放到条件的等待集中。

  1. void await(); 

解除此条件的等待集中所有线程的阻塞状态。

  1. void signalAll(); 

在此条件的等待集中随机选择一个线程,解除其阻塞状态。

  1. void signal(); 


(2) 使用的基本结构:

  1. //初始时ok_to_proceed为false. 
  2. locker.lock() 
  3. try
  4.      while(!ok_to_proceed){ 
  5. //进入等待此条件集中,被阻塞,它维持状态直到另一个线程调用同一个条件上的。 
  6. //signalAll/signal方法时为止。 
  7.        cond.await(); 
  8.      } 
  9. }finally
  10. cker.unlock(); 
  1. locker.lock(); 
  2.    try
  3.       //调用将解除所有等待此条件下的线程的阻塞状态。当线程从等待集中被移走时,它们将再次成为可运行的,调度器将再次激活它们     
  4.       //此时,它们将试图重新进入对象。一旦锁可获得,它们中的某个线程将从await调用返回,从而获得锁并从它被阻塞的地方继续执行。 
  5.       ok_to_proceed = true
  6.       cond.signalAll() or cond.signal(); 
  7.    }finally
  8.        locker.unlock(); 
  9.    } 

ok_to_proceed也是为了防止wait-notify出现的问题,即再wait之间,notify()已经给出通知,此时wait只会一直等待下去,这样就保证了signal()线程的通知被await()线程接收到。

(3) 测试代码:

  1. import java.util.concurrent.locks.Condition; 
  2. import java.util.concurrent.locks.Lock; 
  3. import java.util.concurrent.locks.ReentrantLock; 
  4. class GlobalV{ 
  5.     public final static Lock locker = new ReentrantLock(); 
  6.     public final static Condition cond = locker.newCondition(); 
  7.     public static boolean to_proceed = false
  8.  
  9. class Response extends Thread{ 
  10.     public void run(){ 
  11.         while(true){ 
  12.             GlobalV.locker.lock(); 
  13.             try
  14.                 while(!GlobalV.to_proceed){ 
  15.                     GlobalV.cond.await(); 
  16.                 } 
  17. System.out.println("Response:finish a job"); 
  18.                 GlobalV.to_proceed = false
  19.                  
  20.             }catch(Exception e){ 
  21.                 e.printStackTrace(); 
  22.             }finally
  23.                 GlobalV.locker.unlock(); 
  24.             }    
  25.         } 
  26.     } 
  27.  
  28. class Request extends Thread{ 
  29.     public void run(){ 
  30.         while(true){ 
  31.             GlobalV.locker.lock();   
  32.             try
  33.                 GlobalV.to_proceed = true
  34.                 GlobalV.cond.signalAll(); 
  35.                 System.out.println("Request:send a job to Response");    
  36.             }finally
  37.                 GlobalV.locker.unlock(); 
  38.             } 
  39.             try { 
  40.                 Thread.sleep(2000); 
  41.             } catch (InterruptedException e) { 
  42.                 e.printStackTrace(); 
  43.             } 
  44.         } 
  45.     } 
  46.  
  47. public class Test6 { 
  48.     public static void main(String[] args) { 
  49.         Request req = new Request(); 
  50.         Response res = new Response(); 
  51.         req.start(); 
  52.         res.start(); 
  53.     } 

输出:

Request:send a job to Response
Response:finish a job
Request:send a job to Response
Response:finish a job
Request:send a job to Response
Response:finish a job
Request:send a job to Response
Response:finish a job

#p#

3、读写锁ReentrantReadWriteLock,适用于"读多写少"的多线程应用场景,"读-写"互斥,"写-写"互斥,而读-读可以共享同读锁,即一个线程获取读锁,其它线程可直接进入读,不会被阻塞。

(1) 相关接口

创建读写锁对象

  1. ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock(); 

获取读锁

  1. Lock readLock = rwLock.readLock(); 

获取写锁

  1. Lock writeLock = rwLock.writeLock(); 

(2).读写锁使用基本结构

  1. //对所有的读操作添加读锁 
  2. readLock.lock(); 
  3. try
  4. //code to read 
  5. }finally
  6. readLock.unlock(); 
  7. }  
  1. //对所有的写操作添加写锁 
  2.   writeLock.lock();  
  3.    try{  
  4. //code to write  
  5.    }finally{  
  6.     writeLock.unlock();  
  7.    }  

(3) 测试代码:

  1. import java.util.concurrent.locks.Lock; 
  2. import java.util.concurrent.locks.ReentrantReadWriteLock; 
  3. class Reader extends Thread { 
  4.     private Lock readLock = null
  5.     public Reader(Lock readLock) { 
  6.         this.readLock = readLock; 
  7.     } 
  8.  
  9.     public void run() { 
  10.         while (true) { 
  11.             readLock.lock(); 
  12.             try { 
  13. System.out.println(Thread.currentThread().getName() 
  14.                         + ":read action for 1 seconds-"+ReadWriteLock.testVal); 
  15.             } finally { 
  16.                 readLock.unlock(); 
  17.             } 
  18.             try { 
  19.                 Thread.sleep(1000); 
  20.             } catch (InterruptedException e) { 
  21.                 e.printStackTrace(); 
  22.             } 
  23.         } 
  24.     } 
  25.  
  26. class Writer extends Thread { 
  27.     private Lock writeLock = null
  28.     public Writer(Lock writeLock) { 
  29.         this.writeLock = writeLock; 
  30.     } 
  31.     public void run() { 
  32.         while (true) { 
  33.             writeLock.lock(); 
  34.             try { 
  35. System.out.println(Thread.currentThread().getName() 
  36.                         + ":write action for 2 seconds"); 
  37. if(ReadWriteLock.testVal.equals("1111")) 
  38.                     ReadWriteLock.testVal = "2222"
  39.                 else 
  40.                     ReadWriteLock.testVal = "1111"
  41.             } finally { 
  42.                 writeLock.unlock(); 
  43.             } 
  44.             try { 
  45.                 Thread.sleep(2000); 
  46.             } catch (InterruptedException e) { 
  47.                 e.printStackTrace(); 
  48.             } 
  49.         } 
  50.     } 
  51.  
  52. public class ReadWriteLock { 
  53.     public static String  testVal = "Initiation"
  54.     public static void main(String[] args) { 
  55.         ReentrantReadWriteLock lock = new ReentrantReadWriteLock(); 
  56.         Lock readLock = lock.readLock(); 
  57.         Lock writeLock = lock.writeLock(); 
  58.         Reader reader1 = new Reader(readLock); 
  59.         reader1.setName("reader1"); 
  60.         Reader reader2 = new Reader(readLock); 
  61.         reader2.setName("reader2"); 
  62.         Reader reader3 = new Reader(readLock); 
  63.         reader3.setName("reader3"); 
  64.         Reader reader4 = new Reader(readLock); 
  65.         reader4.setName("reader4"); 
  66.         Writer writer = new Writer(writeLock); 
  67.         writer.setName("writer1"); 
  68.         reader1.start(); 
  69.         reader2.start(); 
  70.         reader3.start(); 
  71.         reader4.start(); 
  72.         writer.start(); 
  73.     } 

输出:

reader1:read action for 1 seconds-Initiation
reader3:read action for 1 seconds-Initiation
writer1:write action for 2 seconds
reader2:read action for 1 seconds-1111
reader4:read action for 1 seconds-1111
reader3:read action for 1 seconds-1111
reader1:read action for 1 seconds-1111
reader4:read action for 1 seconds-1111
reader2:read action for 1 seconds-1111
writer1:write action for 2 seconds
reader4:read action for 1 seconds-2222
reader1:read action for 1 seconds-2222
reader3:read action for 1 seconds-2222
reader2:read action for 1 seconds-2222

4、总结

Lock接口替代synchronized

Lock接口可以比sychronized提供更广泛的锁定操作,可以提供多把不同的锁,且锁之间互不干涉。

Lock接口提供lock()与unlock()方法,使用明确调用来完成同步的,OO思想好于前者。

Lock可以自由操控同步范围(scope)。

Lock接口支持nested lock(嵌套锁定),并提供了丰富的api。

Lock接口提供了tryLock()方法,支持尝试取得某个object lock。

原文链接:http://yuyiming1986.iteye.com/blog/1272979

【编辑推荐】

  1. Java的ClassLoader机制解析
  2. 精解Java中代理模式的实现
  3. Java中的分形几何:把递归用到极致
  4. Java自带的Future多线程模式
  5. 解析Java finally的神秘面纱
责任编辑:林师授 来源: yuyiming1986的博客
相关推荐

2010-03-15 16:31:34

Java多线程

2019-05-27 14:40:43

Java同步机制多线程编程

2017-12-15 10:20:56

MySQLInnoDB同步机制

2009-08-12 13:37:01

Java synchr

2012-07-27 10:02:39

C#

2012-07-09 09:25:13

ibmdw

2010-01-21 11:27:30

linux多线程机制线程同步

2019-08-22 14:30:21

技术Redis设计

2021-10-08 20:30:12

ZooKeeper选举机制

2011-06-22 13:57:54

Java多线程

2011-06-22 13:47:16

Java多线程

2016-09-20 15:21:35

LinuxInnoDBMysql

2019-11-22 18:52:31

进程同步机制编程语言

2017-05-27 20:59:30

Java多线程synchronize

2023-09-01 08:27:34

Java多线程程序

2022-03-24 13:36:18

Java悲观锁乐观锁

2024-02-27 10:44:58

C#线程后端

2020-09-28 06:49:50

Linux系统编程互斥量mutex

2021-03-10 15:59:39

JavaSynchronize并发编程

2009-06-11 17:03:29

Java线程
点赞
收藏

51CTO技术栈公众号