提升spsc_queue - 如何"try pop"?
boost spsc_queue - how to "try pop"?
我有这样的队列:
boost::lockfree::spsc_queue<orders_log, boost::lockfree::capacity<8192>> futOrdersUpdates;
我的问题是,有时我会重复同一项好几次。我添加了一些故障排除代码:
while (bool dequeued = futOrdersUpdates.pop(entryItem) || !endOfTransaction)
{
if (!dequeued) {
dequeueLoger.Debug("no items, try again");
continue;
} else {
if (lastRev != 0 && entryItem.replRev == lastRev) {
dequeueLoger.Debug("duplicate item!");
}
lastRev = entryItem.replRev;
}
// further processing
这个想法是——若并没有设置endOfTransaction
标志,我应该"旋转",否则我可以在队列为空时退出。
在日志中我发现了一些奇怪的东西:
- "没有项目,再试一次"-永远不会出现
- "重复项目!"-IS出现
预期行为:
- 当队列为空但未设置
endOfTransaction
标志时,有时应显示"无项目,重试" - "重复项目!"不应出现
问题是,如果我是"dequee"spsc_queue正确吗?我的代码中有漏洞吗?
我们不可能知道您为什么会得到重复项,因为我们不知道如何填充replRev
。这可能是一场数据竞赛(例如,我希望你不要做lastRev+1
)
-
请注意,您可能永远不会得到"无项目重试",因为循环被完全跳过,例如:
while (bool dequeued = foo()) { assert(dequeued); // can never be false here! }
甚至永远不会进入循环。
-
你得到重复的原因是因为你说
bool dequeued = futOrdersUpdates.pop(entryItem) || !endOfTransaction;
如您所知(请参阅您的评论),这会强制
dequeued
为true
,即使没有项目退出队列,因为endOfTransaction
尚未设置。entryItem
的值是当时未指定的,但可能只包含前一个值,从而导致"重复"消息。
对于我最近给出的使用spsc_queue
的一个不错的演示,请参阅以下答案:动态生成&spsc_queues 的安全使用
相关文章:
- 在c++队列中使用pop和visit实现线程安全
- C++ Python "try: except:"版本
- 编译器是否必须始终删除 try-catch 块(如果它被证明是非抛出的)
- 有没有更好的方法来处理异常? try-catch块真的很丑
- 当无法进行RAII时,如何在C++中"try/finally"?
- C++ push() 和 pop() 方法使用指针的动态 LinkedList 的问题
- C++ 带模板的链表 - 如何创建 pop()?
- 我可以使用 try catch 语句来捕获任何错误而不是具体错误吗?
- std::stack的奇怪行为,pop()返回相同的值
- 为什么以下 POP 功能无法在主机或设备 (CUDA) 上运行?
- 在大型应用程序的main上使用try-catch
- RapidXML 节点在 try catch 块中具有正确的值,但它在块外为 nullptr
- C++ 中类成员的堆栈的 pop() 函数
- 提取 try-catch 时出现运行时错误
- C++:在"try"外部创建的类型会导致错误,但在内部不会
- 没有相应 POP 的编译指示包(推送)会导致堆栈粉碎
- std::unique_ptr 在 try-catch 块中未捕获取消引用异常
- 何时删除 try-catch 块中的指针
- 递归推动POP POSTFIX计算器无法正确执行操作,只需将最后一个数字作为结果
- 提升spsc_queue - 如何"try pop"?