Java中的concurrent包里面的CountDownLatch其实可以把它看作一个计数器,只不过这个计数器的操作是原子操作,同时只能有一个线程去操作这个计数器,也就是同时只能有一个线程去减这个计数器里面的值。
你可以向CountDownLatch对象设置一个初始的数字作为计数值,任何调用这个对象上的await()方法都会阻塞,直到这个计数器的计数值被其他的线程减为0为止。CountDownLatch的一个非常典型的应用场景是:有一个任务想要往下执行,但必须要等到其他的任务执行完毕后才可以继续往下执行。假如我们这个想要继续往下执行的任务调用一个CountDownLatch对象的await()方法,其他的任务执行完自己的任务后调用同一个CountDownLatch对象上的countDown()方法,这个调用await()方法的任务将一直阻塞等待,直到这个CountDownLatch对象的计数值减到0为止。
当中使用CountDownLatch要注意一点,即,需要另外新建一个线程监视工作线程,否则会ANR错误。代码如下:
- public class TestActivity extends Activity {
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- testCountDownLatch();
- }
- private void testCountDownLatch() {
- final Handler handler = new Handler() {
- public void handleMessage(Message msg) {
- if (msg.what == 1){
- }
- }
- };
- new Thread() {
- public void run() {
- CountDownLatch latch = new CountDownLatch(2);// 设置计数值,
- ExecutorService exec = Executors.newCachedThreadPool();// 创建线程池
- exec.execute(new TestRunable1(latch));
- exec.execute(new TestRunable2(latch));
- try {
- latch.await();// 等待所有线程完成操作
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- handler.sendEmptyMessage(1);
- exec.shutdown();
- };
- }.start();
- }
- class TestRunable1 implements Runnable {
- private CountDownLatch latch;
- public TestRunable1(CountDownLatch latch) {
- super();
- this.latch = latch;
- }
- @Override
- public void run() {
- try {
- Thread.sleep(1000 * 5);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- latch.countDown();
- }
- }
- class TestRunable2 implements Runnable {
- private CountDownLatch latch;
- public TestRunable2(CountDownLatch latch) {
- super();
- this.latch = latch;
- }
- @Override
- public void run() {
- try {
- Thread.sleep(1000 * 3);
- latch.countDown();
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- }
- }