多线程(6)同步器

目录

 

一、同步器

二、信号量

三、倒计时门栓

四、障栅

五、交换器

六、同步队列


一、同步器

 java.util.concurrent包包含了几个能帮助人们管理相互合作的线程集的类,见表14-3。这些机制具有为线程之间的共用集节点模式(common rendezvous patterns)提供的“预置功能”(canned functionality)。如果有一个相互合作的线程集满足这些行为模式之一,那么应该直接重用合适的库类而不要试图提供手工的锁与条件的集合。

它能做什么何时使用
Semaphore允许线程集等待直到被允许继续运行为止限制访问资源的线程总数。如果许可数是1,常常阻塞线程直到另一个线程给出许可为止
CountDownLatch允许线程集等待直到计数器减为0当一个或多个线程需要等待直到指定数目的事件发生
CyclicBarrier允许线程集等待直至其中预定数目的线程到达一个公共障栅(barrier),然后可以选择执行一个处理障栅的动作当大量的线程需要在它们的结果可用之前完成时
Exchanger允许两个线程在要交换的对象准备好时交换对象当两个线程工作在同一数据结构的两个实例上的时候,一个向实例添加数据而另一个从实例清除数据
SynchronousQueue允许一个线程把对象交给另一个线程在没有显式同步的情况下,当两个线程准备好将一个对象从一个线程传递到另一个时

二、信号量

概念上讲,一个信号量(semaphore)管理许多的许可证(permits)。为了通过信号量,线程通过调用acquire请求许可。其实没有实际的许可对象,信号量仅维护一个计数。许可的数目是固定的,由此限制了通过的线程数量。其他线程可以通过调用release释放许可。而且,许可不是必须由获取它的线程释放。事实上,任何线程都可以释放任意数目的许可,这可能会增加许可数目以至于超过初始数目。

用法:https://www.cnblogs.com/dolphin0520/p/3920397.html

三、倒计时门栓

 一个倒计时门栓(CountDownLatch)让一个线程集等待直到计数变为0。倒计时门栓是一次性的。—旦计数为0,就不能再重用了。

一个有用的特例是计数值为1的门栓。实现一个只能通过一次的门。线程在门外等候直到另一个线程将计数器值置为0。

举例来讲,假定一个线程集需要一些初始的数据来完成工作。工作器线程被启动并在门外等候。另一个线程准备数据,当数据准备好的时候,调用countDown,所有工作器线程就可以继续运行了。

然后,可以使用第二个门栓检査什么时候所有工作器线程完成工作。用线程数初始化门栓。每个工作器线程在结束前将门栓计数减1。另一个获取工作结果的线程在门外等待,一旦所有工作器线程终止该线程继续运行。

用法:https://www.cnblogs.com/dolphin0520/p/3920397.html

四、障栅

 CyclicBarrier类实现了一个集结点(rendezvous)称为障栅(barrier)。考虑大量线程运行在一次计算的不同部分的情形。当所有部分都准备好时,需要把结果组合在一起。当一个线程完成了它的那部分任务后,我们让它运行到障栅处。一旦所有的线程都到达了这个障栅,障栅就撤销,线程就可以继续运行。

下面是其细节。首先,构造一个障栅,并给出参与的线程数:

CyclicBarrier barrier = new CyclicBarrier(nthreads);

每一个线程做一些工作,完成后在障栅上调用await:    

public void run() {

        doWork();

        barrier.await();

    }

await方法有一个可选的超时参数:

barrier.await(100, TimeUnit.MILLISECONDS);

如果任何一个在障栅上等待的线程离开了障栅,那么障栅就被破坏了(线程可能离开是因为它调用await时设置了超时,或者因为它被中断了)。在这种情况下,所有其他线程的await方法抛出BrokenBarrierException异常。那些已经在等待的线程立即终止await的调用。

 可以提供一个可选的障柵动作(barrier action),当所有线程到达障栅的时候就会执行这一动作。

Runnable barrierAction = ....;

CyclicBarrier barrier = new CyclicBarrier(nthreads, barrierAction);

 该动作可以收集那些单个线程的运行结果

障栅被称为是循环的(cyclic),因为可以在所有等待线程被释放后被重用。在这一点上,有别于CountDownLatch,CountDownLatch只能被使用一次

用法:https://www.cnblogs.com/dolphin0520/p/3920397.html

五、交换器

当两个线程在同一个数据缓冲区的两个实例上工作的时候,就可以使用交换器。典型的情况是,一个线程向缓冲区填入数据,另一个线程消耗这些数据。当它们都完成以后,相互交换缓冲区。

用法:https://www.cnblogs.com/nayitian/archive/2013/09/12/3317384.html

六、同步队列

同步队列是一种将生产者与消费者线程配对的机制。当一个线程调用SynchronousQueue的put方法时,它会阻塞直到另一个线程调用take方法为止,反之亦然。与Exchanger的情况不同,数据仅仅沿一个方向传递,从生产者到消费者。

即使SynchronousQueue类实现了BlockingQueue接口,概念上讲,它依然不是一个队列它没有包含任何元素它的size方法总是返回0

用法及原理:https://www.cnblogs.com/duanxz/p/3252267.html

 

参考:《Java核心技术 卷I》

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值