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

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();
  6. }

这种结构保证在任何时刻只有一个线程能够进入临界区,如果一个线程锁住了锁对象,其他任何线程在调用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. class WorkerOne extends Thread{
  4.     private Lock locker;
  5.     public WorkerOne (Lock locker){
  6.         this.locker = locker;
  7.     }
  8.     
  9.     public void run(){
  10.         locker.lock();
  11.         try{
  12. System.out.println(Thread.currentThread().getName()+":step into critical section");
  13.         }finally{
  14.             locker.unlock();    
  15.         }
  16.     }
  17. }
  18. class WorkerTwo extends Thread{
  19.     private Lock locker;
  20.     public WorkerTwo (Lock locker){
  21.         this.locker = locker;
  22.     }
  23.     
  24.     public void sayHello(){
  25.         locker.lock();
  26.         try{    System.out.println(Thread.currentThread().getName()+":call sayHello()");
  27.             Thread.sleep(1000);
  28.         } catch (InterruptedException e) {
  29.             e.printStackTrace();
  30.         }finally{
  31.             locker.unlock();
  32.         }
  33.     }
  34.     
  35.     public void run(){
  36.         locker.lock();  
  37.         try{        System.out.println(Thread.currentThread().getName()+":setp into critical section");
  38.                         //测试锁的可重入性 
  39.             sayHello();
  40.         }finally{
  41.             locker.unlock();    
  42.         }
  43.     }
  44. }
  45. public class Test5 {
  46.     public static void main(String[] args) {
  47.         Lock locker = new ReentrantLock();
  48.         WorkerOne wo= new WorkerOne(locker);
  49.         wo.setName("WorkerOne");
  50.         WorkerTwo wt = new WorkerTwo(locker);
  51.         wt.setName("WorkerTwo");
  52.         
  53.         wt.start();
  54.         wo.start(); 
  55.     }
  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();
  11. }
 
 
 
  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.     }
  54. }

输出:

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

分享标题:去故就新Java线程新同步机制
浏览路径:http://www.shufengxianlan.com/qtweb/news24/366074.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联