1. 程式人生 > >CountDownLatch 門閂 例項詳解

CountDownLatch 門閂 例項詳解

CountDownLatch 介紹

CountDownLatch是JAVA提供在java.util.concurrent包下的一個輔助類,可以把它看成是一個計數器,其內部維護著一個count計數,只不過對這個計數器的操作都是原子操作,同時只能有一個執行緒去操作這個計數器。如果呼叫物件上的await()方法,呼叫者就會一直阻塞,直到別人呼叫countDown方法,將計數減到0,才可以繼續執行。

CountDownLatch可以避免等待佇列,它可以讓程式碼執行效率更高一些。

程式碼例項

final CountDownLatch latch = new CountDownLatch(1); //定義一個門閂(計數器),一旦定義,不可變。

門閂常用方法:

latch.await();                        等待門閂的開放,不是進入等待佇列。

latch.countDown();               呼叫此方法,門閂減一

以一個案例講解CountDownLatch門閂

要求:定義一個容器,該容器最大限度為10,當容器元素新增到第5個的時候,列印容器大小。

程式碼:

/**
 * CountDownLatch 門閂
 */
package concurrent.t02;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;

public class Test_03 {
	public static void main(String[] args) {
		final Test_03_Container t = new Test_03_Container();
		final CountDownLatch latch = new CountDownLatch(1);

		new Thread(new Runnable(){
			@Override
			public void run() {
				if(t.size() != 5){
					try {
						latch.await(); // 等待門閂的開放。 不是進入等待佇列
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
				}
				System.out.println("size = 5");
			}
		}).start();
		
		new Thread(new Runnable() {
			@Override
			public void run() {
				for(int i = 0; i < 10; i++){
					System.out.println("add Object to Container " + i);
					t.add(new Object());
					if(t.size() == 5){
						latch.countDown(); // 門閂-1
					}
					try {
						TimeUnit.SECONDS.sleep(1);
					} catch (Exception e) {
						e.printStackTrace();
					}
				}
			}
		}).start();
	}
}

class Test_03_Container{
	List<Object> container = new ArrayList<>();
	
	public void add(Object o){
		this.container.add(o);
	}
	
	public int size(){
		return this.container.size();
	}
}

門閂應用場景:馬拉松比賽