前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Lock和Synchronizer是什么关系?

Lock和Synchronizer是什么关系?

作者头像
dhyuan
发布2022-05-30 14:16:09
3380
发布2022-05-30 14:16:09
举报
文章被收录于专栏:响应式编程

引子:书上说AbstractQueuedSynchronizer(AQS)是构建锁和Synchronizer的框架。锁,好像大家都知道,至少自以为都知道 :)。那什么是synchronizer呢?Lock和Synchronize是什么关系?

Synchronizer,它是一个根据自身状态调节线程执行的对象。就是用来协调(多)线程执行的对象。从这个角度讲Java的内置锁就是一种synchronizer,它以互斥的可重入的方式协调/控制线程的执行。

Java的阻塞队列也是一种synchronizser。

再比如:信号量Semaphore、闭锁Latch、关卡Barrier都是不同类型的synchronizer。

信号量

Semaphore可以用来控制同时访问某资源的线程数量。把对这些资源的获取操作包装起来,获取资源前先调用信号量的acquire()申请许可,资源使用使用完毕后通过release()释放许可。对池化资源的管理一般可用信号量完成。

闭锁

英文Latch有门闩的意思,门闩就是用来把门关紧不让出入。在并发编程里也是这个意思,我们用latch这个对象禁止线程的执行,什么时候允许线程通过这个门闩呢?对于CountDownLatch来说,就是计数变为0的时候。对于FutureTask来说就是可以拿到计算结果的时候(当然也可能是计算异常了)。一个应用场景:可以把闭锁当作一个发令枪,它可以让线程等到信号后一起运行。

看!还真有这种闩数大于一的门闩。

关卡

Barrier中文就是“障碍物、栅栏”的意思,文绉绉的翻译就是“关卡”。它其实像极了闭锁,与闭锁不同的是:闭锁等待的是事件,而关卡等待的是线程。从API调用的感觉感觉上讲,Barrier是在工作线程正儿八经的工作都执行完毕后(取决于业务场景),调用barrier.await()使工作线程阻塞住,直到所有其它工作线程也都完成个各自的任务并都调用了barrier.await(),这个时候关卡就被冲破。这里等待的条件是有足够的线程调用barrier.await()。而Latch是在工作线程中调用latch.await(),等待闭锁被开启来执行。闭锁是怎样被开启的?它是依靠latch.countDown()到0后被开启的,谁countdown跟哪个线程没什么直接关系。而barrier.await()是需要实实在在的线程阻塞,这就是为什么说barrier等待的是线程,latch等的是信号。

经常使用关卡到场景是,一个任务分成n个子任务后,等待这n个子任务都完成后再做下一步工作。

CyclicBarrier的构造函数接受一个线程数以及一个关卡被突破后要执行的动作。

Exchanger是另一种关卡,用来为两个线程交换数据,当然是以线程安全的方式交换。

Reference:

https://web.mit.edu/6.005/www/fa15/classes/23-locks/

《Java并发编程实践》

https://en.wikipedia.org/wiki/Synchronization_(computer_science)

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-03-30,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 响应式编程 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档