成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

靈魂拷問,AQS 是個啥???

開發 開發工具
AQS 隊列在內部維護了一個 FIFO 的雙向鏈表,如果對數據結構比較熟的話,應該很容易就能想到,在雙向鏈表中,每個節點都有兩個指針,分別指向直接前驅節點和直接后繼節點。

 啥?你連 AQS 是啥都不知道?

如果想要精通 Java 并發的話, AQS 是一定要掌握的。今天跟著阿粉一起搞一搞。

[[328494]]

基本概念

AQS 是 AbstractQueuedSynchronizer 的簡稱,翻譯成中文就是 抽象隊列同步器 ,這三個單詞分開來看:

  • Abstract (抽象):也就是說, AQS 是一個抽象類,只實現一些主要的邏輯,有些方法推遲到子類實現
  • Queued (隊列):隊列有啥特征呢?先進先出( FIFO )對吧?也就是說, AQS 是用先進先出隊列來存儲數據的
  • Synchronizer (同步):即 AQS 實現同步功能

以上概括一下, AQS 是一個用來構建鎖和同步器的框架,使用 AQS 能簡單而又高效地構造出同步器。

AQS 內部實現

AQS 隊列在內部維護了一個 FIFO 的雙向鏈表,如果對數據結構比較熟的話,應該很容易就能想到,在雙向鏈表中,每個節點都有兩個指針,分別指向直接前驅節點和直接后繼節點。使用雙向鏈表的優點之一,就是從任意一個節點開始都很容易訪問它的前驅節點和后繼節點。

在 AQS 中,每個 Node 其實就是一個線程封裝,當線程在競爭鎖失敗之后,會封裝成 Node 加入到 AQS 隊列中;獲取鎖的線程釋放鎖之后,會從隊列中喚醒一個阻塞的 Node (也就是線程)

AQS 使用 volatile 的變量 state 來作為資源的標識:

  1. private volatile int state; 

關于 state 狀態的讀取與修改,子類可以通過覆蓋 getState() 和 setState() 方法來實現自己的邏輯,其中比較重要的是:

  1. // 傳入期望值 expect ,想要修改的值 update ,然后通過 Unsafe 的 compareAndSwapInt() 即 CAS 操作來實現 
  2. protected final boolean compareAndSetState(int expect, int update) { 
  3.     // See below for intrinsics setup to support this 
  4.     return unsafe.compareAndSwapInt(this, stateOffset, expect, update); 

下面是 AQS 中兩個重要的成員變量:

  1. private transient volatile Node head;   // 頭結點 
  2. private transient volatile Node tail;   // 尾節點 

關于 AQS 維護的雙向鏈表,在源碼中是這樣解釋的:

  • The wait queue is a variant of a "CLH" (Craig, Landin, and Hagersten) lock queue. CLH locks are normally used for spinlocks. We instead use them for blocking synchronizers, but use the same basic tactic of holding some of the control information about a thread in the predecessor of its node.

也就是 AQS 的等待隊列是 “CLH” 鎖定隊列的變體

直接來一張圖會更形象一些:

Node 節點維護的是線程,控制線程的一些操作,具體來看看是 Node 是怎么做的:

  1. static final class Node { 
  2.     /** Marker to indicate a node is waiting in shared mode */ 
  3.     // 標記一個節點,在 共享模式 下等待 
  4.     static final Node SHARED = new Node(); 
  5.   
  6.     /** Marker to indicate a node is waiting in exclusive mode */ 
  7.     // 標記一個節點,在 獨占模式 下等待 
  8.     static final Node EXCLUSIVE = null
  9.  
  10.     /** waitStatus value to indicate thread has cancelled */ 
  11.     // waitStatus 的值,表示該節點從隊列中取消 
  12.     static final int CANCELLED =  1; 
  13.   
  14.     /** waitStatus value to indicate successor's thread needs unparking */ 
  15.     // waitStatus 的值,表示后繼節點在等待喚醒 
  16.     // 只有處于 signal 狀態的節點,才能被喚醒 
  17.     static final int SIGNAL    = -1; 
  18.   
  19.     /** waitStatus value to indicate thread is waiting on condition */ 
  20.     // waitStatus 的值,表示該節點在等待一些條件 
  21.     static final int CONDITION = -2; 
  22.   
  23.  /** 
  24.      * waitStatus value to indicate the next acquireShared should 
  25.      * unconditionally propagate 
  26.     */ 
  27.     // waitStatus 的值,表示有資源可以使用,新 head 節點需要喚醒后繼節點 
  28.     // 如果是在共享模式下,同步狀態應該無條件傳播下去 
  29.     static final int PROPAGATE = -3; 
  30.  
  31.  // 節點狀態,取值為 -3,-2,-1,0,1 
  32.     volatile int waitStatus; 
  33.  
  34.  // 前驅節點 
  35.     volatile Node prev; 
  36.  
  37.  // 后繼節點 
  38.     volatile Node next
  39.  
  40.  // 節點所對應的線程 
  41.     volatile Thread thread; 
  42.  
  43.  // condition 隊列中的后繼節點 
  44.     Node nextWaiter; 
  45.  
  46.  // 判斷是否是共享模式 
  47.     final boolean isShared() { 
  48.         return nextWaiter == SHARED; 
  49.     } 
  50.  
  51.  /** 
  52.  * 返回前驅節點 
  53.  */ 
  54.     final Node predecessor() throws NullPointerException { 
  55.         Node p = prev; 
  56.         if (p == null
  57.             throw new NullPointerException(); 
  58.         else 
  59.             return p; 
  60.     } 
  61.  
  62.     Node() {    // Used to establish initial head or SHARED marker 
  63.     } 
  64.  
  65.  /** 
  66.  * 將線程構造成一個 Node 節點,然后添加到 condition 隊列中 
  67.  */ 
  68.     Node(Thread thread, Node mode) {     // Used by addWaiter 
  69.         this.nextWaiter = mode; 
  70.         this.thread = thread; 
  71.     } 
  72.  
  73.  /** 
  74.  * 等待隊列用到的方法 
  75.  */ 
  76.     Node(Thread thread, int waitStatus) { // Used by Condition 
  77.         this.waitStatus = waitStatus; 
  78.         this.thread = thread; 
  79.     } 

AQS 如何獲取資源

在 AQS 中,獲取資源的入口是 acquire(int arg) 方法,其中 arg 是獲取資源的個數,來看下代碼:

  1. public final void acquire(int arg) { 
  2.     if (!tryAcquire(arg) && 
  3.         acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) 
  4.         selfInterrupt(); 

在獲取資源時,會首先調用 tryAcquire 方法,這個方法是在子類中具體實現的

如果通過 tryAcquire 獲取資源失敗,接下來會通過 addWaiter(Node.EXCLUSIVE) 方法,將這個線程插入到等待隊列中,具體代碼:

  1. private Node addWaiter(Node mode) { 
  2.  // 生成該線程所對應的 Node 節點 
  3.     Node node = new Node(Thread.currentThread(), mode); 
  4.     // 將 Node 插入到隊列中 
  5.     Node pred = tail; 
  6.     if (pred != null) { 
  7.         node.prev = pred; 
  8.         // 使用 CAS 操作,如果成功就返回 
  9.         if (compareAndSetTail(pred, node)) { 
  10.             pred.next = node; 
  11.             return node; 
  12.         } 
  13.     } 
  14.     // 如果 pred == null 或者 CAS 操作失敗,則調用 enq 方法再次自旋插入 
  15.     enq(node); 
  16.     return node; 
  17.   
  18. // 自旋 CAS 插入等待隊列 
  19. private Node enq(final Node node) { 
  20.     for (;;) { 
  21.         Node t = tail; 
  22.         if (t == null) { // Must initialize 
  23.             if (compareAndSetHead(new Node())) 
  24.                 tail = head; 
  25.         } else { 
  26.             node.prev = t; 
  27.             if (compareAndSetTail(t, node)) { 
  28.                 t.next = node; 
  29.                 return t; 
  30.             } 
  31.         } 
  32.     } 

在上面能夠看到使用的是 CAS 自旋插入,這是因為在 AQS 中會存在多個線程同時競爭資源的情況,進而一定會出現多個線程同時插入節點的操作,這里使用 CAS 自旋插入是為了保證操作的線程安全性

現在呢,申請 acquire(int arg) 方法,然后通過調用 addWaiter 方法,將一個 Node 插入到了隊列尾部。處于等待隊列節點是從頭結點開始一個一個的去獲取資源,獲取資源方式如下:

  1. final boolean acquireQueued(final Node node, int arg) { 
  2.     boolean failed = true
  3.     try { 
  4.         boolean interrupted = false
  5.         for (;;) { 
  6.             final Node p = node.predecessor(); 
  7.             // 如果 Node 的前驅節點 p 是 head,說明 Node 是第二個節點,那么它就可以嘗試獲取資源 
  8.             if (p == head && tryAcquire(arg)) { 
  9.              // 如果資源獲取成功,則將 head 指向自己 
  10.                 setHead(node); 
  11.                 p.next = null; // help GC 
  12.                 failed = false
  13.                 return interrupted; 
  14.             } 
  15.             // 節點進入等待隊列后,調用 shouldParkAfterFailedAcquire 或者 parkAndCheckInterrupt 方法 
  16.             // 進入阻塞狀態,即只有頭結點的線程處于活躍狀態 
  17.             if (shouldParkAfterFailedAcquire(p, node) && 
  18.                 parkAndCheckInterrupt()) 
  19.                 interrupted = true
  20.         } 
  21.     } finally { 
  22.         if (failed) 
  23.             cancelAcquire(node); 
  24.     } 

在獲取資源時,除了 acquire 之外,還有三個方法:

  • acquireInterruptibly :申請可中斷的資源(獨占模式)
  • acquireShared :申請共享模式的資源
  • acquireSharedInterruptibly :申請可中斷的資源(共享模式)

到這里,關于 AQS 如何獲取資源就說的差不多了,接下來看看 AQS 是如何釋放資源的

AQS 如何釋放資源

釋放資源相對于獲取資源來說,簡單了很多。源碼如下:

  1. public final boolean release(int arg) { 
  2.  // 如果釋放鎖成功 
  3.     if (tryRelease(arg)) {  
  4.      // 獲取 AQS 隊列中的頭結點 
  5.         Node h = head; 
  6.         // 如果頭結點不為空,且狀態 != 0 
  7.         if (h != null && h.waitStatus != 0) 
  8.          // 調用 unparkSuccessor(h) 方法,喚醒后續節點 
  9.             unparkSuccessor(h); 
  10.         return true
  11.     } 
  12.     return false
  13.  
  14. private void unparkSuccessor(Node node) { 
  15.     int ws = node.waitStatus; 
  16.     // 如果狀態是負數,嘗試將它改為 0 
  17.     if (ws < 0) 
  18.         compareAndSetWaitStatus(node, ws, 0); 
  19.  // 得到頭結點的后繼節點 
  20.     Node s = node.next
  21.     // 如果 waitStatus 大于 0 ,說明這個節點被取消 
  22.     if (s == null || s.waitStatus > 0) { 
  23.         s = null
  24.         // 那就從尾節點開始,找到距離 head 最近的一個 waitStatus<=0 的節點進行喚醒 
  25.         for (Node t = tail; t != null && t != node; t = t.prev) 
  26.             if (t.waitStatus <= 0) 
  27.                 s = t; 
  28.     } 
  29.     // 如果后繼節點不為空,則將其從阻塞狀態變為非阻塞狀態 
  30.     if (s != null
  31.         LockSupport.unpark(s.thread); 

AQS 兩種資源共享模式

資源有兩種共享模式:

  • 獨占模式( Exclusive ):資源是獨占的,也就是一次只能被一個線程占有,比如 ReentrantLock
  • 共享模式( Share ):同時可以被多個線程獲取,具體的資源個數可以通過參數來確定,比如 Semaphore/CountDownLatch

看到這里, AQS 你 get 了嘛?

 

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2019-11-19 10:32:55

Java語言程序員

2020-05-29 11:48:01

安全運維信息安全網絡安全

2022-12-12 08:46:11

2020-05-22 08:13:45

敏捷開發OKR

2022-08-26 01:10:32

TCPSYNLinux

2022-03-16 18:27:39

開發低代碼軟件開發

2019-08-12 11:14:00

JVM垃圾對象

2019-07-29 10:10:06

Java內存線程安全

2021-05-26 05:22:48

SQL 數據庫SELECT

2023-06-16 14:10:00

TCPUDP網絡通信

2021-03-12 09:24:58

Redis面試場景

2019-08-01 10:20:10

2022-05-30 18:37:03

數據個人信息人工智能

2019-10-09 10:00:02

集群故障場景

2019-10-09 09:39:15

PythonHDFS大數據

2014-11-26 14:46:47

代碼

2021-06-02 09:47:48

RSA2021

2023-02-13 13:37:20

ChatGPTAI微軟

2021-02-03 10:17:03

5G5G專網通信市場

2023-05-04 12:12:00

ChatGPTAI
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩在线不卡 | 久久99久久99精品免视看婷婷 | 国产传媒在线观看 | 成人免费网站 | 色综合av| av黄色免费 | 成人国产在线视频 | 欧美三级电影在线播放 | 亚洲第一在线视频 | 在线看av网址 | 成人高清在线视频 | 97超碰中文网| 国产精品久久久久久久久久久免费看 | 日韩第一区 | 精品欧美久久 | 国产成人a亚洲精品 | 久草www| 日本天天操 | 国产一区二区精华 | 国产一区二区三区在线 | 99这里只有精品视频 | 国产在线视频一区二区董小宛性色 | 国产精品免费在线 | 国产精品一区二区av | 亚洲精品大全 | 亚洲视频欧美视频 | 欧美黄色网络 | 先锋资源在线 | 亚洲一区二区久久久 | 中文字幕一区二区三区不卡 | 国产福利在线 | 欧洲亚洲视频 | 亚洲最色视频 | 国产精品久久久亚洲 | 天天操天天射天天舔 | 亚洲精品在线免费观看视频 | 欧美男人天堂 | 免费黄色a级毛片 | 亚洲 中文 欧美 日韩 在线观看 | 日本精品免费在线观看 | 欧美激情精品久久久久久免费 |