自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

去故就新 Java線程新同步機制

開發(fā) 后端
新的Java線程同步機制,主要對可重入鎖ReentrantLock,條件對象Condition以及讀寫鎖ReentrantReadWriteLock機制解析。

1、可重入鎖ReentrantLock,相當(dāng)于synchronized塊,為臨界區(qū)提供互斥訪問機制。

(1) 相關(guān)的接口

創(chuàng)建一個可重入鎖

  1. Lock lock = new ReentrantLock(); 

請求鎖,如果鎖被當(dāng)前另一個線程持有,則阻塞。

  1. void lock();

釋放鎖

  1. void unlock(); 

非阻塞型lock()

  1. boolean tryLock(); 


(2) 使用基本結(jié)構(gòu)

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

這種結(jié)構(gòu)保證在任何時刻只有一個線程能夠進(jìn)入臨界區(qū),如果一個線程鎖住了鎖對象,其他任何線程在調(diào)用lock時,都會被阻塞,直到第一個線程釋放鎖對象。而且無論try塊是否拋出異常,都會執(zhí)行finally block,解鎖locker。

(3) 鎖的可重入性

鎖是可重入的,線程能夠重復(fù)地獲取它已經(jīng)擁有的鎖。鎖對象維護(hù)一個持有計數(shù)(hold count)來追蹤對lock方法的嵌套調(diào)用。線程在每次調(diào)用lock后都要調(diào)用unlock來釋放鎖。由于這個特性,被一個鎖保護(hù)的代碼可以調(diào)用另一個使用相同鎖的方法。

(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,相當(dāng)于wait-notify機制,提供一種線程間的等待通知機制,condition中的等待-通知方法是await(),signal(),signalAll(),也需要在互斥環(huán)境下被調(diào)用。

(1) 相關(guān)的接口

創(chuàng)建Condition對象,Condition對象是跟Lock關(guān)聯(lián)在一起的。

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

把此線程放到條件的等待集中。

  1. void await(); 

解除此條件的等待集中所有線程的阻塞狀態(tài)。

  1. void signalAll(); 

在此條件的等待集中隨機選擇一個線程,解除其阻塞狀態(tài)。

  1. void signal(); 


(2) 使用的基本結(jié)構(gòu):

  1. //初始時ok_to_proceed為false. 
  2. locker.lock() 
  3. try
  4.      while(!ok_to_proceed){ 
  5. //進(jìn)入等待此條件集中,被阻塞,它維持狀態(tài)直到另一個線程調(diào)用同一個條件上的。 
  6. //signalAll/signal方法時為止。 
  7.        cond.await(); 
  8.      } 
  9. }finally
  10. cker.unlock(); 
  1. locker.lock(); 
  2.    try
  3.       //調(diào)用將解除所有等待此條件下的線程的阻塞狀態(tài)。當(dāng)線程從等待集中被移走時,它們將再次成為可運行的,調(diào)度器將再次激活它們     
  4.       //此時,它們將試圖重新進(jìn)入對象。一旦鎖可獲得,它們中的某個線程將從await調(diào)用返回,從而獲得鎖并從它被阻塞的地方繼續(xù)執(zhí)行。 
  5.       ok_to_proceed = true
  6.       cond.signalAll() or cond.signal(); 
  7.    }finally
  8.        locker.unlock(); 
  9.    } 

ok_to_proceed也是為了防止wait-notify出現(xiàn)的問題,即再wait之間,notify()已經(jīng)給出通知,此時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,適用于"讀多寫少"的多線程應(yīng)用場景,"讀-寫"互斥,"寫-寫"互斥,而讀-讀可以共享同讀鎖,即一個線程獲取讀鎖,其它線程可直接進(jìn)入讀,不會被阻塞。

(1) 相關(guān)接口

創(chuàng)建讀寫鎖對象

  1. ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock(); 

獲取讀鎖

  1. Lock readLock = rwLock.readLock(); 

獲取寫鎖

  1. Lock writeLock = rwLock.writeLock(); 

(2).讀寫鎖使用基本結(jié)構(gòu)

  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、總結(jié)

Lock接口替代synchronized

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

Lock接口提供lock()與unlock()方法,使用明確調(diào)用來完成同步的,OO思想好于前者。

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

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

Lock接口提供了tryLock()方法,支持嘗試取得某個object lock。

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

【編輯推薦】

  1. Java的ClassLoader機制解析
  2. 精解Java中代理模式的實現(xiàn)
  3. Java中的分形幾何:把遞歸用到極致
  4. Java自帶的Future多線程模式
  5. 解析Java finally的神秘面紗
責(zé)任編輯:林師授 來源: yuyiming1986的博客
相關(guān)推薦

2024-07-05 08:32:36

2010-03-15 16:31:34

Java多線程

2024-06-28 08:45:58

2019-05-27 14:40:43

Java同步機制多線程編程

2017-12-15 10:20:56

MySQLInnoDB同步機制

2024-07-08 12:51:05

2009-08-12 13:37:01

Java synchr

2012-07-27 10:02:39

C#

2012-07-09 09:25:13

ibmdw

2025-03-31 00:01:12

2010-01-21 11:27:30

linux多線程機制線程同步

2019-08-22 14:30:21

技術(shù)Redis設(shè)計

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

進(jìn)程同步機制編程語言

2024-07-25 11:53:53

2017-05-27 20:59:30

Java多線程synchronize

2023-09-01 08:27:34

Java多線程程序
點贊
收藏

51CTO技術(shù)棧公眾號