java semaphore 等待时间_一个java同步工具类Semaphore的详解
Semaphore是java并發包里面的一個工具類,我們限制可以訪問某些資源的線程數目就可以使用Semaphore了。這篇文章將對Semaphore的概念和使用進行一個詳解。
一、概念理解
官方是這樣解釋的:
Semaphore用于限制可以訪問某些資源(物理或邏輯的)的線程數目,他維護了一個許可證集合,有多少資源需要限制就維護多少許可證集合,假如這里有N個資源,那就對應于N個許可證,同一時刻也只能有N個線程訪問。一個線程獲取許可證就調用acquire方法,用完了釋放資源就調用release方法。
不過這樣的解釋實在有點抽象,現在用我自己的話來解釋一下:
相信在學生時代都去餐廳打過飯,假如有3個窗口可以打飯,同一時刻也只能有3名同學打飯。第四個人來了之后就必須在外面等著,只要有打飯的同學好了,就可以去相應的窗口了。
比如說這張圖,就全是了Semaphore的基本使用。認識一個知識點的最好方式就是直接去使用,我們干脆直接上代碼來看看如何使用。
二、代碼使用
這個案例使用的就是我們之前的小例子,也就是去餐廳打飯的案例。
我們先看Test類:
1public?class?SemaphoreTest{
2????//第一步:定義一個信號量Semaphore
3????static?Semaphore?sp?=?new?Semaphore(3);
4????public?static?void?main(String[]?args){
5????????//第二步:定義10個學生去打飯
6????????for(int?i=0;i<10;i++)?{
7????????????//十個學生用一個信號量
8????????????new?Student(sp,?"學生"+i).start();
9????????}
10????}
11}
在這個代碼中我們看到,主要是new了一個Semaphore,然后賦給每一位同學Student,接下來我們就來好好看看Student線程是如何實現的。
1public?class?Student?extends?Thread{
2????private?Semaphore?sp?=null;
3????private?String?name?=?null;
4????public?Student(Semaphore?sp,?String?name){
5????????this.sp?=?sp;
6????????this.name?=?name;
7????}
8????@Override
9????public?void?run(){
10????????try?{
11????????????sp.acquire();
12????????????System.out.println(name+"拿到了打飯的許可");
13????????????TimeUnit.SECONDS.sleep(3);
14????????}?catch?(InterruptedException?e)?{
15????????????e.printStackTrace();
16????????}finally?{
17????????????System.out.println(name+"打好了飯,釋放這個窗口");
18????????????sp.release();
19????????}
20????}
21}
在這個Student類中我們最主要看run方法的實現,首先我們通過acquire獲取了當前窗口的許可,然后休眠3秒代表打飯,最后在finally使用release方法釋放這個窗口許可證。代碼很簡單,原理很清楚,我們測試一波:
這個結果你也看到了,基本上同一時刻只能有三個學生在窗口旁邊。
在這里你可能有一個疑問了,Semaphore好像和synchronized關鍵字沒什么區別,都可以實現同步,如果是這樣那說明我們還沒有真正理解jdk的注釋,他只是限制了訪問某些資源的線程數,其實并沒有實現同步,我們可以看一下:
1????@Override
2????public?void?run(){
3????????try?{
4????????????System.out.println(name+"進入了餐廳");
5????????????sp.acquire();
6????????????System.out.println(name+"拿到了打飯的許可");
7????????????TimeUnit.SECONDS.sleep(3);
8????????}?catch?(InterruptedException?e)?{
9????????????e.printStackTrace();
10????????}finally?{
11????????????System.out.println(name+"打好了飯,釋放這個窗口");
12????????????sp.release();
13????????}
14????}
現在我們在獲取許可前增加了一條輸出語句,也就是能打印出有哪個線程進入了,再去測試一波:
結果很清晰,所以對于Semaphore來說,我們需要記住的其實是資源的互斥而不是資源的同步,在同一時刻是無法保證同步的,但是卻可以保證資源的互斥。
三、其他方法
在上面我們使用最基本的acquire方法和release方法就可以實現Semaphore最常見的功能,不過其他方法還是需要我們去了解一下的。
1、acquire(int permits)
從此信號量獲取給定數目的許可,在提供這些許可前一直將線程阻塞,或者線程已被中斷。就好比是一個學生占兩個窗口。這同時也對應了相應的release方法。
2、release(int permits)
釋放給定數目的許可,將其返回到信號量。這個是對應于上面的方法,一個學生占幾個窗口完事之后還要釋放多少
3、availablePermits()
返回此信號量中當前可用的許可數。也就是返回當前還有多少個窗口可用。
4、reducePermits(int reduction)
根據指定的縮減量減小可用許可的數目。
5、hasQueuedThreads()
查詢是否有線程正在等待獲取資源。
6、getQueueLength()
返回正在等待獲取的線程的估計數目。該值僅是估計的數字。
7、tryAcquire(int permits, ?long timeout, TimeUnit unit)
如果在給定的等待時間內此信號量有可用的所有許可,并且當前線程未被中斷,則從此信號量獲取給定數目的許可。
8、acquireUninterruptibly(int permits)
從此信號量獲取給定數目的許可,在提供這些許可前一直將線程阻塞。
基本上常見的使用方法都在這,Semaphore底層是由AQS和Uasafe完成的,篇幅問題在這里不贅述了。感謝各位支持。
推薦閱讀:
總結
以上是生活随笔為你收集整理的java semaphore 等待时间_一个java同步工具类Semaphore的详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Uzi绝境奇迹未能再现!TES 3:2
- 下一篇: 苹果又被告了!1500名英国开发者索赔7