我知道文档说明该对象是线程安全的,但这是否意味着从所有方法对其进行的所有访问都是线程安全的?因此,如果我put()
一次从多个线程中调用它,并且一次take()
在同一实例上调用它,会不会发生什么不好的事情?
快速答案是肯定的,它们是线程安全的。但是不要让它在那里…
首先,一个小的内部管理BlockingQueue
是一个接口,任何不是线程安全的实现都将破坏书面合同。您包括的链接是指LinkedBlockingQueue
,它具有一定的灵巧性。
您包含的链接引起了一个有趣的观察,是的,其中有两个锁LinkedBlockingQueue
。但是,它无法理解,实际上正在处理“简单”实现会被犯规的极端情况,这就是为什么take和put方法比起初期望的要复杂的原因。
LinkedBlockingQueue
进行了优化,以避免在读取和写入时使用相同的锁,这减少了争用,但是对于正确的行为,它依赖于队列不为空。当队列中包含元素时,推送和弹出点不在内存的同一区域,可以避免争用。但是,当队列为空时,则无法避免争用,因此需要额外的代码来处理这种常见的“边缘”情况。这是代码复杂度和性能/可伸缩性之间的常见折衷。
接下来的问题是,如何LinkedBlockingQueue
知道队列何时为空/不为空,从而处理线程呢?答案是,它使用AtomicInteger
和Condition
作为两个额外的并发数据结构。所述AtomicInteger
用于检查队列的长度是否是零,并且条件是用于等待一个信号时,队列可能是所希望的状态,以通知一个等待的线程。这种额外的协调确实会产生开销,但是在测量中已显示,在增加并发线程数时,此技术的开销低于使用单个锁引入的争用。
下面,我从复制了代码,LinkedBlockingQueue
并添加了解释它们如何工作的注释。在较高级别上,take()
首先将所有其他呼叫锁定take()
,然后put()
根据需要发出信号。
put()
以类似的方式工作,首先阻止所有其他呼叫put()
,然后take()
在必要时发出信号。
从put()
方法:
// putLock coordinates the calls to put() only; further coordination
// between put() and take() follows below
putLock.lockInterruptibly();
try {
// block while the queue is full; count is shared between put() and take()
// and is safely visible between cores but prone to change between calls
// a while loop is used because state can change between signals, which is
// why signals get rechecked and resent.. read on to see more of that
while (count.get() == capacity) {
notFull.await();
}
// we know that the queue is not full so add
enqueue(e);
c = count.getAndIncrement();
// if the queue is not full, send a signal to wake up
// any thread that is possibly waiting for the queue to be a little
// emptier -- note that this is logically part of 'take()' but it
// has to be here because take() blocks itself
if (c + 1 < capacity)
notFull.signal();
} finally {
putLock.unlock();
}
if (c == 0)
signalNotEmpty();
从 take()
takeLock.lockInterruptibly();
try {
// wait for the queue to stop being empty
while (count.get() == 0) {
notEmpty.await();
}
// remove element
x = dequeue();
// decrement shared count
c = count.getAndDecrement();
// send signal that the queue is not empty
// note that this is logically part of put(), but
// for thread coordination reasons is here
if (c > 1)
notEmpty.signal();
} finally {
takeLock.unlock();
}
if (c == capacity)
signalNotFull();
问题内容: 我一直在假设线程安全也不是线程安全,但是在最近的一次讨论中,一位同事告诉我线程安全。 因此,我做了一些研究,却一无所获。很多人认为它是线程安全的,很多人认为它不是线程安全的。而且,最重要的是,文档没有以一种或另一种方式说任何话,不是为了,甚至不是。 那是什么呢? 问题答案: 这是指向Java 7 中Calendar和GregorianCalendar的源代码的链接。 如果阅读该代码,您
我有一个在Oracle 11g DB上运行的insert语句,如下所示: 这里有一个处理PostgreSQL的类似问题。但是,由于Oracle序列由所有会话共享,所以我不能相信DB会给出当前会话中最后插入的值。
问题内容: 我有多个线程试图更新MySQL数据库?使用executeUpdate方法是线程安全的吗? 问题答案: 不,使用它不是线程安全的。 实际上,如果其他某个线程使用一条语句,然后另一个线程调用executeUpdate(),则另一个线程的s(如果有的话)将被关闭。“ javadoc.sql.Statement的JavaDoc(PreparedStatement是其子类型) ”如果存在打开的语
有一次,我被印上了“祝贺”,有一次,我被印上了“站台”。
我们正在尝试使用以下查询将数据流式传输到postgres 11: 基本上“在表中插入记录,如果它已经存在 - 我们希望将此查询连接到消息队列,并在多个实例中的高并发环境中运行它。使用此查询,可能会从不同的连接访问同一行。对我们来说,只有具有最高交付时间戳的项目最终才能进入表是至关重要的 根据文件: 在冲突时,DO UPDATE保证原子插入或更新结果;如果没有独立的错误,那么即使在高并发的情况下,这
如果这个家伙不是线程安全的,是否有类似的东西(可能来自Apache,...)? 谢了。