当前位置: 首页 > 知识库问答 >
问题:

对可重入锁优于同步块的质疑

梁宏才
2023-03-14

我正在阅读Java中重入锁和同步块之间的比较。我正在浏览互联网上的各种资源。我在同步块上使用Reentrant锁发现的一个缺点是,在上一个中,您必须显式使用try final block来调用final telock在final块中获取的锁上调用解锁方法,因为您的关键代码部分可能会引发异常,并且可能会造成大麻烦, 如果线程没有释放锁,而在后一个中,JVM本身负责在发生异常时释放锁。

我不太相信这个缺点,因为使用try last没什么大不了的block.As我们已经在ex(流关闭等)中使用了很长时间。有人能告诉我重入锁在同步块上的其他一些缺点吗?

共有2个答案

丌官哲彦
2023-03-14

对于低或最小的争用,synchronized几乎总是更快,因为它允许JVM执行一些优化,例如偏置锁定锁定省略等。以下是它的工作原理的更多详细信息:

让我们假设某个监视器由线程A持有,线程B请求这个监视器。在这种情况下,监视器将把它的状态更改为膨胀。简而言之,这意味着所有试图获取这个监视器的线程都将在操作系统级别等待设置,这是相当昂贵的。

现在,如果线程A在线程B请求之前释放了监视器,那么所谓的< code>rebias操作将由廉价的(在现代CPU上)< code >比较和交换操作来执行。

现在让我们来看看< code>ReentrantLock。每个线程调用< code>lock()或< code > lock interruptible()方法都会导致通过< code>CAS操作进行锁定尝试。

结论:在低争用情况下,最好使用< code >html" target="_blank">同步。在高度争用的情况下,最好使用< code>ReentrantLock。对于介于两者之间的所有情况,很难说清楚,请考虑执行基准测试以找出哪个解决方案更快。

闾丘书
2023-03-14

对于不同的用例,ReentrantLock是不同的工具。虽然您可以使用这两种方法来解决大多数同步问题(这就是它们的用途),但是它们有不同的优缺点。

Synchronized最简单:你写synchronized,仅此而已。对于现代的JVM来说,这是相当快的速度,但是它的缺点是,无论实际上是否需要,所有试图进入同步块的线程都会被挂起。如果您过于频繁地使用synchronized,这会极大地降低多线程的速度,最糟糕的情况是单线程的执行速度会更快。

由于线程问题仅在有人正在写入而其他人正在读取/写入相同的数据部分时发生,因此程序经常会遇到问题,即它们理论上可以在没有同步的情况下运行,因为大多数线程只是读取,但是偶尔会有一个写入,它强制执行同步块。这就是锁的用途:您可以更好地控制实际同步的时间。

除了构造函数中的一个公平参数之外,基本的ReentantLock允许您决定何时释放锁,并且您可以在多个点上执行,这样就可以在最适合您的时候执行。它的其他变体,如ReentantReadWriteLock,允许您进行许多不同步的读取,除非有写入。缺点是这在Java代码中得到了解决,这使得它明显比“本机”同步块慢。也就是说:只有在您知道使用此锁的优化增益大于损失的情况下,您才应该使用它。

在正常情况下,只有当你实际监控它时,你才能分辨速度的差异,通过运行分析器以复杂的方式检查前后的速度。

 类似资料:
  • 在示例代码中 在这个页面上, lock1和lock2分别控制c1和c2上的更新。 然而, 正在获取对象lock1的锁并在同步块时释放它 被执行。 当这个代码块被执行时,这个对象的成员c1上可能还有一个更新——我看不出这个更新是如何被代码中的lock1上的同步所阻止的。 只有对象lock1可以独占访问——除此之外别无它物(?) 那么,实施情况如何 在上面的代码中不同于 甚至 当c1是一个对象而不是一

  • 问题内容: 我正在建模一个游戏,其中多个玩家(线程)同时移动。玩家当前所在位置的信息被存储两次:该玩家具有一个变量“ hostField”,该变量引用板上的一个字段,每个字段都有一个ArrayList,用于存储当前位于该字段的玩家。 我对拥有冗余信息的事实不是很满意,但是我发现如果不遍历大数据集就无法避免这种情况。 但是,当玩家从一个字段移到另一个字段时,我想确保(1)冗余信息保持链接(2)此刻没

  • 问题内容: 输出是 线程1开始 线程1的内部演示 线程2开始 线程2的内部演示 由于,执行尚未结束。 我已经通过类的同一个实例来的两个类的构造函数和。 对in 的调用位于一个块中。 要在通话中是 没有 的块。 所以,当被执行时,由于块,的监视器应锁定,导致拒绝访问,以由所述。 但是,这没有发生。 预期的输出是 (输出在完成之前。) 因此,我的基本问题是: 即使 尚未完成 块, 如何成功执行? 问题

  • 我对Java同步()块的理解是,如果一个线程已经拥有一个对象上的锁,它可以进入一个在同一个对象上同步的不同块(重入同步)。下面,我相信JVM使用引用计数来增加/减少线程获得锁的次数,并且锁只有在计数为零时才会释放。 所以我的问题是,如果你遇到这样的代码: 当调用etc()时,具体会发生什么?它仅仅是减少计数,还是不顾计数释放锁? 在第一种情况下,在我看来,如果发生了锁重新进入,它将产生死锁,因为它

  • 我们知道,有一个最大可重入限制:;块也有可重入限制吗? 更新:我发现很难为同步可重入编写测试代码: 有人能帮我写一些同步可重入极限测试的代码吗?

  • 为什么java可重入锁不会导致死锁?