- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我有一个写入和读取 ConcurrentLinkedQueue 的多线程应用程序,它在概念上用于支持列表/表中的条目。我最初为此使用了 ConcurrentHashMap,效果很好。一项新要求需要跟踪进入的订单条目,因此可以根据某些条件以最旧的顺序删除它们。 ConcurrentLinkedQueue 似乎是一个不错的选择,而且在功能上它运行良好。
可配置数量的条目保存在内存中,当达到限制时提供新条目时,将以最早的优先顺序搜索队列以查找可以删除的条目。某些条目不会被系统删除并等待客户端交互。
似乎正在发生的事情是我在发生的队列前面有一个条目,比如 100K 条目之前。该队列似乎配置的条目数量有限(size() == 100),但在分析时,我发现内存中有约 100K 个 ConcurrentLinkedQueue$Node 对象。这似乎是设计使然,只需看一眼 ConcurrentLinkedQueue 的源代码,删除操作只会删除对正在存储的对象的引用,但会保留链表以供迭代。
最后我的问题是:是否有一种“更好”的懒惰方式来处理这种性质的集合?我喜欢 ConcurrentLinkedQueue 的速度,我只是无法承受在这种情况下似乎可能出现的无限泄漏。如果没有,我似乎必须创建第二个结构来跟踪订单并且可能会遇到相同的问题,加上同步问题。
最佳答案
这里实际发生的是 remove 方法准备一个轮询线程来清空链接引用。
ConcurrentLinkedQueue 是一个非阻塞线程安全队列实现。但是,当您尝试从队列中轮询节点时,它是一个具有两个功能的过程。首先将值置空,然后将引用置空。 CAS 操作是单个原子函数,不会为轮询提供即时解决方案。
轮询时发生的事情是,第一个成功的线程将获取节点的值并将该值清空,然后该线程将尝试清空引用。另一个线程可能会进入并尝试从队列中进行轮询。为确保此 Queue 拥有非阻塞属性(即一个线程的失败不会导致另一个线程的失败),新的传入线程将查看该值是否为 null,如果为 null,则该线程将使该引用为 null 并尝试再次轮询()。
所以您在这里看到的是删除线程只是准备任何新的轮询线程以使引用为空。尝试实现非阻塞删除功能我认为几乎是不可能的,因为这需要三个原子功能。值的 null 是对所述节点的 null 引用,最后是从该节点的父节点到其后继节点的新引用。
回答你的最后一个问题。不幸的是,没有更好的方法来实现删除和维护队列的非阻塞状态。至少在这一点上是这样。一旦处理器开始推出 2 路和 3 路套管,那么这是可能的。
关于java - ConcurrentLinkedQueue$Node 在 remove() 之后保留在堆中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2547161/
我正在使用 .remove() 方法删除一个 html 元素,同时对于这个元素,我有一个事件处理程序,但它没有被触发。为什么会这样呢?这是jsFiddle和代码:HTML Delete I'm goi
所以我尝试从另一篇文章中编写此代码: while(fscanf(orderFile," %49[^;];%d; %49[^\n]",fileName,&seconds,timeValue) == 3)
我正在阅读 Nicolai M.Josuttis 撰写的“The C++ STL. A Tutorial and References”一书,在专门介绍 STL 算法的一章中,作者陈述如下:如果你调用
是否有一种简单的机制来确定 DownloadManager remove() 何时完成,因为它看起来是部分异步的。该函数几乎立即返回下载表中已删除的条目计数,但实际的文件系统管理似乎被插入了某个后台线
我愿意: getActionBarToolbar().removeView(logoImage); getActionBarToolbar().addView(logoImage, lp); 我得到:
我有类(class)评论一对多关系。在类(class)表中有 id 和 title 列。在 Review 表中,有 id、comment 和 course_id,其中“course_id”作为指向类(
我在 stackoverflow 上阅读了不同的答案,了解如何销毁 wigdet/jQueryObject 并取消绑定(bind)其上的所有事件。 这就是我的想法。 $('选择器').remove()
我有一个由一个线程填充的 byte[] 列表,然后我有另一个线程正在从该列表中读取并通过网络发送项目。 每次我读取线程 2 中的项目时,我都想将其从内存中清除。但是因为我正在使用线程,如果我使用 .r
就算法而言,从连续数组中删除一组元素可以分两部分有效地完成。 将所有不删除的元素移到数组的前面。 将数组标记得更小。 这可以在 C++ 中使用 erase-remove 习惯用法来完成。 vector
我尝试删除包含在 map 中渲染的制造商的 View 。当我单击按钮时,事件 click .ver 被激活,但没有任何反应,并且我收到以下错误:Uncaught TypeError: undefine
场景: 使用 jQuery 2.0.1 构建的应用程序。 您的团队更喜欢原生 JavaScript。 选项有jQuery .remove()和 ChildNode.remove() . 您需要删除节点
最初我有一个像这样的删除功能: function ViewWorkflowDetail(btn, workflowId) { $("#workflowDetailPanel").remov
我正在编写 C++ 代码来解决 Leetcode 中的这个问题:https://leetcode.com/problems/remove-element/ Given an array nums an
根据太阳, "Iterator.remove is the only safe way to modify a collection during iteration; the behavior is
众所周知,从 std::vector 中完全删除所需项的一种好方法是 erase-remove idiom . 如以上链接中所述(截至本文发布日期),在代码中,erase-remove 习惯用法如下所
我在 HashSet 上调用 Iterator.remove() 时遇到问题。 我有一组带有时间戳的对象。在将新项目添加到集合之前,我会遍历集合,识别该数据对象的旧版本并将其删除(在添加新对象之前)。
这段代码: Collection col = new ArrayList(); col.add("a"); col.add("b"); col.add("c");
我试图通过在下面输入来卸载 conda 环境基础, conda env remove -n base 正如我所建议的那样,我尝试通过使用来停用基地 conda deactivate base 我再次尝
我已经对我的 IOS 应用程序进行了质量扫描分析。我收到以下警告: The binary has Runpath Search Path (@rpath) set. In certain cases
这个问题已经有答案了: Properly removing an Integer from a List (8 个回答) 已关闭 4 年前。 我是java新手。看起来很简单,但我不明白为什么会发生这种
我是一名优秀的程序员,十分优秀!