欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java并發(fā)編程Semaphore計(jì)數(shù)信號(hào)量詳解

 更新時(shí)間:2017年10月25日 10:35:12   作者:IAMTJW  
這篇文章主要介紹了Java并發(fā)編程Semaphore計(jì)數(shù)信號(hào)量詳解,具有一定參考價(jià)值,需要的朋友可以了解下。

Semaphore 是一個(gè)計(jì)數(shù)信號(hào)量,它的本質(zhì)是一個(gè)共享鎖。信號(hào)量維護(hù)了一個(gè)信號(hào)量許可集。線程可以通過調(diào)用acquire()來獲取信號(hào)量的許可;當(dāng)信號(hào)量中有可用的許可時(shí),線程能獲取該許可;否則線程必須等待,直到有可用的許可為止。 線程可以通過release()來釋放它所持有的信號(hào)量許可(用完信號(hào)量之后必須釋放,不然其他線程可能會(huì)無法獲取信號(hào)量)。

簡單示例:

package me.socketthread;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
public class SemaphoreLearn { 
  //信號(hào)量總數(shù) 
  private static final int SEM_MAX = 12; 
  public static void main(String[] args) {  
    Semaphore sem = new Semaphore(SEM_MAX); 
    //創(chuàng)建線程池 
    ExecutorService threadPool = Executors.newFixedThreadPool(3); 
    //在線程池中執(zhí)行任務(wù) 
    threadPool.execute(new MyThread(sem, 7)); 
    threadPool.execute(new MyThread(sem, 4)); 
    threadPool.execute(new MyThread(sem, 2)); 
    //關(guān)閉池 
    threadPool.shutdown(); 
  } 
} 
  class MyThread extends Thread { 
    private volatile Semaphore sem;  // 信號(hào)量 
    private int count;    // 申請(qǐng)信號(hào)量的大小  
     
    MyThread(Semaphore sem, int count) { 
      this.sem = sem; 
      this.count = count; 
    } 
    public void run() { 
      try { 
       // 從信號(hào)量中獲取count個(gè)許可 
        sem.acquire(count); 
        Thread.sleep(2000); 
        System.out.println(Thread.currentThread().getName() + " acquire count="+count); 
      } catch (InterruptedException e) { 
        e.printStackTrace(); 
      } finally { 
        // 釋放給定數(shù)目的許可,將其返回到信號(hào)量。 
        sem.release(count); 
        System.out.println(Thread.currentThread().getName() + " release " + count + ""); 
      } 
    } 
  } 

執(zhí)行結(jié)果:

pool-1-thread-2 acquire count=4
pool-1-thread-1 acquire count=7
pool-1-thread-1 release 7
pool-1-thread-2 release 4
pool-1-thread-3 acquire count=2
pool-1-thread-3 release 2

線程1和線程2會(huì)并發(fā)執(zhí)行,因?yàn)閮烧叩男盘?hào)量和沒有超過總信號(hào)量,當(dāng)前兩個(gè)線程釋放掉信號(hào)量之后線程3才能繼續(xù)執(zhí)行。

源碼分析:

1、構(gòu)造函數(shù)

在構(gòu)造函數(shù)中會(huì)初始化信號(hào)量值,這值最終是作為鎖標(biāo)志位state的值

Semaphore sem = new Semaphore(12);//簡單來說就是給鎖標(biāo)識(shí)位state賦值為12 

2、Semaphore.acquire(n);簡單理解為獲取鎖資源,如果獲取不到線程阻塞

Semaphore.acquire(n);//從鎖標(biāo)識(shí)位state中獲取n個(gè)信號(hào)量,簡單來說是state = state-n 此時(shí)state大于0表示可以獲取信號(hào)量,如果小于0則將線程阻塞 
public void acquire(int permits) throws InterruptedException { 
    if (permits < 0) throw new IllegalArgumentException(); 
    //獲取鎖 
    sync.acquireSharedInterruptibly(permits); 
  } 

acquireSharedInterruptibly中的操作是獲取鎖資源,如果可以獲取則將state= state-permits,否則將線程阻塞

public final void acquireSharedInterruptibly(int arg) 
      throws InterruptedException { 
    if (Thread.interrupted()) 
      throw new InterruptedException(); 
    if (tryAcquireShared(arg) < 0)//tryAcquireShared中嘗試獲取鎖資源 
      doAcquireSharedInterruptibly(arg); //將線程阻塞 
  } 

tryAcquireShared中的操作是嘗試獲取信號(hào)量值,簡單來說就是state=state-acquires ,如果此時(shí)小于0則返回負(fù)值,否則返回大于新值,再判斷是否將當(dāng)線程線程阻塞

protected int tryAcquireShared(int acquires) { 
      for (;;) { 
        if (hasQueuedPredecessors()) 
          return -1; 
      //獲取state值 
        int available = getState(); 
      //從state中獲取信號(hào)量 
        int remaining = available - acquires; 
        if (remaining < 0 || 
          compareAndSetState(available, remaining)) 
        //如果信號(hào)量小于0則直接返回,表示無法獲取信號(hào)量,否則將state值修改為新值 
          return remaining; 
      } 
    } 

doAcquireSharedInterruptibly中的操作簡單來說是將當(dāng)前線程添加到FIFO隊(duì)列中并將當(dāng)前線程阻塞。

/會(huì)將線程添加到FIFO隊(duì)列中,并阻塞  
private void doAcquireSharedInterruptibly(int arg)  
    throws InterruptedException {  
    //將線程添加到FIFO隊(duì)列中  
    final Node node = addWaiter(Node.SHARED);  
    boolean failed = true;  
    try {  
      for (;;) {  
        final Node p = node.predecessor();  
        if (p == head) {  
          int r = tryAcquireShared(arg);  
          if (r >= 0) {  
            setHeadAndPropagate(node, r);  
            p.next = null; // help GC  
            failed = false;  
            return;  
          }  
        }  
        //parkAndCheckInterrupt完成線程的阻塞操作  
        if (shouldParkAfterFailedAcquire(p, node) &&  
          parkAndCheckInterrupt())  
          throw new InterruptedException();  
      }  
    } finally {  
      if (failed)  
        cancelAcquire(node);  
    }  
  } 

3、Semaphore.release(int permits),這個(gè)函數(shù)的實(shí)現(xiàn)操作是將state = state+permits并喚起處于FIFO隊(duì)列中的阻塞線程。

public void release(int permits) { 
    if (permits < 0) throw new IllegalArgumentException(); 
  //state = state+permits,并將FIFO隊(duì)列中的阻塞線程喚起 
    sync.releaseShared(permits); 
  } 

releaseShared中的操作是將state = state+permits,并將FIFO隊(duì)列中的阻塞線程喚起。

public final boolean releaseShared(int arg) { 
    //tryReleaseShared將state設(shè)置為state = state+arg 
    if (tryReleaseShared(arg)) { 
      //喚起FIFO隊(duì)列中的阻塞線程 
      doReleaseShared(); 
      return true; 
    } 
    return false; 
  } 

tryReleaseShared將state設(shè)置為state = state+arg

protected final boolean tryReleaseShared(int releases) { 
      for (;;) { 
        int current = getState(); 
        int next = current + releases; 
        if (next < current) // overflow 
          throw new Error("Maximum permit count exceeded"); 
        //將state值設(shè)置為state=state+releases 
        if (compareAndSetState(current, next)) 
          return true; 
      } 
    } 

doReleaseShared()喚起FIFO隊(duì)列中的阻塞線程

private void doReleaseShared() {  
  
    for (;;) {  
      Node h = head;  
      if (h != null && h != tail) {  
        int ws = h.waitStatus;  
        if (ws == Node.SIGNAL) {  
          if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))  
            continue;      // loop to recheck cases  
          //完成阻塞線程的喚起操作  
          unparkSuccessor(h);  
        }  
        else if (ws == 0 &&  
             !compareAndSetWaitStatus(h, 0, Node.PROPAGATE))  
          continue;        // loop on failed CAS  
      }  
      if (h == head)          // loop if head changed  
        break;  
    }  
  }  

總結(jié):Semaphore簡單來說設(shè)置了一個(gè)信號(hào)量池state,當(dāng)線程執(zhí)行時(shí)會(huì)從state中獲取值,如果可以獲取則線程執(zhí)行,并且在執(zhí)行后將獲取的資源返回到信號(hào)量池中,并喚起其他阻塞線程;如果信號(hào)量池中的資源無法滿足某個(gè)線程的需求則將此線程阻塞。

Semaphore源碼:

public class Semaphore implements java.io.Serializable { 
  private static final long serialVersionUID = -3222578661600680210L; 
  private final Sync sync; 
  abstract static class Sync extends AbstractQueuedSynchronizer { 
    private static final long serialVersionUID = 1192457210091910933L; 
    //設(shè)置鎖標(biāo)識(shí)位state的初始值 
    Sync(int permits) { 
      setState(permits); 
    } 
    //獲取鎖標(biāo)識(shí)位state的值,如果state值大于其需要的值則表示鎖可以獲取 
    final int getPermits() { 
      return getState(); 
    } 
    //獲取state值減去acquires后的值,如果大于等于0則表示鎖可以獲取 
    final int nonfairTryAcquireShared(int acquires) { 
      for (;;) { 
        int available = getState(); 
        int remaining = available - acquires; 
        if (remaining < 0 || 
          compareAndSetState(available, remaining)) 
          return remaining; 
      } 
    } 
    //釋放鎖 
    protected final boolean tryReleaseShared(int releases) { 
      for (;;) { 
        int current = getState(); 
        //將state值加上release值 
        int next = current + releases; 
        if (next < current) // overflow 
          throw new Error("Maximum permit count exceeded"); 
        if (compareAndSetState(current, next)) 
          return true; 
      } 
    } 
    //將state的值減去reductions 
    final void reducePermits(int reductions) { 
      for (;;) { 
        int current = getState(); 
        int next = current - reductions; 
        if (next > current) // underflow 
          throw new Error("Permit count underflow"); 
        if (compareAndSetState(current, next)) 
          return; 
      } 
    } 
    final int drainPermits() { 
      for (;;) { 
        int current = getState(); 
        if (current == 0 || compareAndSetState(current, 0)) 
          return current; 
      } 
    } 
  } 
  //非公平鎖 
  static final class NonfairSync extends Sync { 
    private static final long serialVersionUID = -2694183684443567898L; 
    NonfairSync(int permits) { 
      super(permits); 
    } 
    protected int tryAcquireShared(int acquires) { 
      return nonfairTryAcquireShared(acquires); 
    } 
  } 
  //公平鎖 
  static final class FairSync extends Sync { 
    private static final long serialVersionUID = 2014338818796000944L; 
    FairSync(int permits) { 
      super(permits); 
    } 
    protected int tryAcquireShared(int acquires) { 
      for (;;) { 
        if (hasQueuedPredecessors()) 
          return -1; 
        int available = getState(); 
        int remaining = available - acquires; 
        if (remaining < 0 || 
          compareAndSetState(available, remaining)) 
          return remaining; 
      } 
    } 
  } 
  //設(shè)置信號(hào)量 
  public Semaphore(int permits) { 
    sync = new NonfairSync(permits); 
  } 
  public Semaphore(int permits, boolean fair) { 
    sync = fair ? new FairSync(permits) : new NonfairSync(permits); 
  } 
  //獲取鎖 
  public void acquire() throws InterruptedException { 
    sync.acquireSharedInterruptibly(1); 
  } 
  public void acquireUninterruptibly() { 
    sync.acquireShared(1); 
  } 
  public boolean tryAcquire() { 
    return sync.nonfairTryAcquireShared(1) >= 0; 
  } 
  public boolean tryAcquire(long timeout, TimeUnit unit) 
    throws InterruptedException { 
    return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout)); 
  } 
  public void release() { 
    sync.releaseShared(1); 
  } 
  //獲取permits值鎖 
  public void acquire(int permits) throws InterruptedException { 
    if (permits < 0) throw new IllegalArgumentException(); 
    sync.acquireSharedInterruptibly(permits); 
  } 
  public void acquireUninterruptibly(int permits) { 
    if (permits < 0) throw new IllegalArgumentException(); 
    sync.acquireShared(permits); 
  } 
  public boolean tryAcquire(int permits) { 
    if (permits < 0) throw new IllegalArgumentException(); 
    return sync.nonfairTryAcquireShared(permits) >= 0; 
  } 
  public boolean tryAcquire(int permits, long timeout, TimeUnit unit) 
    throws InterruptedException { 
    if (permits < 0) throw new IllegalArgumentException(); 
    return sync.tryAcquireSharedNanos(permits, unit.toNanos(timeout)); 
  } 
  //釋放 
  public void release(int permits) { 
    if (permits < 0) throw new IllegalArgumentException(); 
    sync.releaseShared(permits); 
  } 
  public int availablePermits() { 
    return sync.getPermits(); 
  } 
  public int drainPermits() { 
    return sync.drainPermits(); 
  } 
  protected void reducePermits(int reduction) { 
    if (reduction < 0) throw new IllegalArgumentException(); 
    sync.reducePermits(reduction); 
  } 
  public boolean isFair() { 
    return sync instanceof FairSync; 
  } 
  public final boolean hasQueuedThreads() { 
    return sync.hasQueuedThreads(); 
  } 
  public final int getQueueLength() { 
    return sync.getQueueLength(); 
  } 
  protected Collection<Thread> getQueuedThreads() { 
    return sync.getQueuedThreads(); 
  } 
  public String toString() { 
    return super.toString() + "[Permits = " + sync.getPermits() + "]"; 
  } 
} 

總結(jié)

以上就是本文關(guān)于Java并發(fā)編程Semaphore計(jì)數(shù)信號(hào)量詳解的全部內(nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:Java并發(fā)編程之重入鎖與讀寫鎖、Java系統(tǒng)的高并發(fā)解決方法詳解java高并發(fā)鎖的3種實(shí)現(xiàn)示例代碼等,有什么問題,可以留言交流討論。感謝朋友們對(duì)本站的支持!

相關(guān)文章

  • Spring的Ioc模擬實(shí)現(xiàn)詳細(xì)介紹

    Spring的Ioc模擬實(shí)現(xiàn)詳細(xì)介紹

    這篇文章主要介紹了Spring的Ioc模擬實(shí)現(xiàn)詳細(xì)介紹,具有一定參考價(jià)值,需要的朋友可以了解下。
    2017-11-11
  • Idea如何關(guān)閉或開啟引用提示Usages和Annotations

    Idea如何關(guān)閉或開啟引用提示Usages和Annotations

    這篇文章主要介紹了Idea如何關(guān)閉或開啟引用提示Usages和Annotations問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • Java單例模式與破壞單例模式概念原理深入講解

    Java單例模式與破壞單例模式概念原理深入講解

    單例模式(Singleton?Pattern)是?Java?中最簡單的設(shè)計(jì)模式之一。這種類型的設(shè)計(jì)模式屬于創(chuàng)建型模式,它提供了一種創(chuàng)建對(duì)象的最佳方式。這種模式涉及到一個(gè)單一的類,該類負(fù)責(zé)創(chuàng)建自己的對(duì)象,同時(shí)確保只有單個(gè)對(duì)象被創(chuàng)建
    2023-02-02
  • Java的Spring框架中AOP項(xiàng)目的一般配置和部署教程

    Java的Spring框架中AOP項(xiàng)目的一般配置和部署教程

    這篇文章主要介紹了Java的Spring框架中AOP項(xiàng)目的一般配置和部署教程,AOP面向方面編程的項(xiàng)目部署結(jié)構(gòu)都比較類似,因而也被看作是Spring的一種設(shè)計(jì)模式使用,需要的朋友可以參考下
    2016-04-04
  • 如何解決Java多線程死鎖問題

    如何解決Java多線程死鎖問題

    死鎖是一個(gè)很嚴(yán)重的、必須要引起重視的問題,本文主要介紹了死鎖的定義,解決方法和面試會(huì)遇到的問題,感興趣的可以了解一下
    2021-05-05
  • Java執(zhí)行SQL腳本文件到數(shù)據(jù)庫詳解

    Java執(zhí)行SQL腳本文件到數(shù)據(jù)庫詳解

    這篇文章主要為大家詳細(xì)介紹了Java執(zhí)行SQL腳本文件到數(shù)據(jù)庫的相關(guān)方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Java Hutool(糊涂)工具類索引詳解

    Java Hutool(糊涂)工具類索引詳解

    這篇文章主要介紹了Java Hutool(糊涂)工具類索引,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-07-07
  • SpringBoot整合RestTemplate用法的實(shí)現(xiàn)

    SpringBoot整合RestTemplate用法的實(shí)現(xiàn)

    本篇主要介紹了RestTemplate中的GET,POST,PUT,DELETE、文件上傳和文件下載6大常用的功能,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-08-08
  • Java如何使用while循環(huán)計(jì)算一個(gè)整數(shù)的位數(shù)

    Java如何使用while循環(huán)計(jì)算一個(gè)整數(shù)的位數(shù)

    這篇文章主要介紹了Java使用while循環(huán)計(jì)算一個(gè)整數(shù)的位數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • 深入理解java long 存儲(chǔ)時(shí)間戳

    深入理解java long 存儲(chǔ)時(shí)間戳

    存儲(chǔ)時(shí)間打算用時(shí)間戳來存儲(chǔ),打算用long類型來代表時(shí)間戳,這篇文章主要介紹了深入理解java long 存儲(chǔ)時(shí)間戳,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2018-10-10

最新評(píng)論