您的位置:首页 >SynchronousQueue是什么?无缓冲队列详解
发布于2026-02-22 阅读(0)
扫一扫,手机访问
SynchronousQueue 的 size() 总是 0,因为它不存储元素,仅作为线程间直接交接的同步点,put/take 必须配对阻塞等待,无缓冲区,故 size() 恒为 0、isEmpty() 恒为 true,peek() 和 iterator() 均抛 UnsupportedOperationException。

因为 SynchronousQueue 根本不存数据——它不是“队列”,而是“交接点”。你调用 put(),线程就卡住,直到另一个线程正好在调 take();反过来也一样。中间没有缓冲区,没有数组,没有链表节点存着你的元素。所以 size() 永远返回 0,isEmpty() 永远是 true,连 peek() 和 iterator() 都直接抛 UnsupportedOperationException。
offer(e, timeout, unit) 期望“尽力插入”——它和 put(e) 行为一致:超时前必须等到消费者,否则返回 falsesize == 0 别慌,这是正常态,不是 bugExecutors.newCachedThreadPool() 内部用的就是 SynchronousQueue,核心就一条:任务来了,有空闲线程就立刻交过去;没有,就新建线程。它不希望任务排队等——等就意味着延迟、积压、内存占用。
LinkedBlockingQueue(哪怕 capacity=1),任务就会先入队,再等线程来取,破坏“直交”语义SynchronousQueue 强制把背压转成拒绝策略(如 AbortPolicy),更可控TransferStack)可能导致新任务优先被最新创建的空闲线程抢走,而不是等待最久的线程——对低延迟敏感场景反而更优构造时传 true 就是公平模式(用 TransferQueue),false 或无参就是非公平(TransferStack)。这不是“好不好”的问题,而是“要不要保序”。
put 的等得久,优先被后续 take 接走;适合需要确定性调度的场景,比如信号量式同步、严格 FIFO 的事件分发put 可能立刻匹配上前一个刚 take 完正在自旋的线程,吞吐略高,但顺序不可预测;newCachedThreadPool 默认用它,因为任务本身无序,快比稳重要最典型的错误是写这种逻辑:queue.offer(task) || fallbackToDisk(),以为 offer() 失败才走备选——但 SynchronousQueue.offer() 只有在没线程等着 take() 时才返回 false,且不阻塞。这跟 put() 的“必须等到”完全不同,极易引发逻辑错乱。
offer(e, 0, TimeUnit.NANOSECONDS),而非无参 offer()put() 再 take()——必然死锁,因为没人配合交接put() 或 take(),第一反应不该是扩容,而是查消费者线程是否挂了、阻塞了、或根本没启动
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9