- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
比较 removeAll(Collection<?> c)
的速度我觉得很有趣在 Collection
中声明的调用.现在我知道微基准测试很难正确执行,我不会考虑几毫秒的差异,但我相信我的结果是有效的,因为我反复运行它们并且它们非常可重现。
假设我有两个不太小的集合,比如说 100,000 个连续的整数元素,而且它们大部分重叠,例如左边有 5,000 个,右边没有。现在我只需调用:
left.removeAll(right);
当然这一切都取决于左右集合的类型。如果正确的集合是 HashMap ,速度会非常快,因为这是完成查找的地方。但仔细观察,我注意到两个无法解释的结果。我用 ArrayList
尝试了所有测试这是排序的,另一个是洗牌的(使用 Collections.shuffle()
,如果这很重要的话)。
第一个奇怪的结果是:
00293 025% shuffled ArrayList, HashSet
00090 008% sorted ArrayList, HashSet
现在要么从排序的 ArrayList
中删除元素比从随机列表中删除或从 HashSet
中查找连续值更快比查找随机值更快。
现在是另一个:
02311 011% sorted ArrayList, shuffled ArrayList
01401 006% sorted ArrayList, sorted ArrayList
现在这表明在排序的 ArrayList
中查找(对左侧列表的每个元素使用 contains()
调用)比随机列表更快。现在,如果我们可以利用它已排序的事实并使用二进制搜索,那将非常容易,但我不这样做。
这两个结果对我来说都很神秘。我无法通过查看代码或我的数据结构知识来解释它们。它与处理器缓存访问模式有什么关系吗? JIT 编译器是否优化了一些东西?但如果是这样,哪个?我进行了热身并连续运行了几次测试,但也许我的基准测试存在根本问题?
最佳答案
性能差异的原因是内存访问模式:访问内存中连续的元素比进行随机内存访问更快(由于内存预取、cpu 缓存等)
当您最初填充集合时,您会在内存中按顺序创建所有元素,因此当您遍历它(foreach、removeAll 等)时,您正在访问缓存友好的连续内存区域。当你打乱集合时——元素在内存中保持相同的顺序,但指向这些元素的指针不再是相同的顺序,所以当你遍历集合时,你将访问例如第 10 个、第 1 个、然后是第 5 个元素,它对缓存非常不友好并且会破坏性能。
您可以查看此问题,其中更详细地显示了此效果: Why filtering an unsorted list is faster than filtering a sorted list
关于Java 性能 : Search and removal speed on removeAll(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29626953/
我不确定 groovy 中的 removeAll 是如何工作的,但我希望它会返回 [40289454470ea94601470ea977d00018] def list = ['40289454470
我在使用 Netbeans 制作的 Java 应用程序中遇到了奇怪的问题。这是一款内存卡游戏。主类是一个 JFrame,它保存相同大小的面板。面板通过网格布局进行分割,每个单元格都包含一张卡片,另一个
List vAllBatchList = getAllBatchCollection().toList(); //Has 700k records List vKeepableBatchCollect
我编写了一个通用的 Partition 类(分区是将一个集合划分为不相交的子集,称为部分)。在内部这是一个 Map和一个 Map> ,其中整数是零件的标签。例如partition.getLabel(T
我有两个自创建类对象的数组列表。比较后,我从两者中删除了公共(public)对象,并准备了获取公共(public)元素的方法。找到公共(public)元素后,我通过调用 removeAll() 方法删
我想要编写一段代码,它接受一个列表列表,将其拆分为 9 个子列表,并从每个子列表中的所有列表中删除数字。但是,当我的代码运行时,它会从所有列表中删除数字,而不仅仅是从原始列表中获取的部分 for (i
我是 Linq 的新手。根据我的理解,LINQ 应该只用于查询而不用于修改集合或数据库等。 如果是这样,微软为什么要提供 RemoveAll() 扩展? 据我所知,RemoveAll() 修改集合。
有人可以解释为什么以下内容无法按我的预期工作吗? 按下“应该”按钮会导致显示仅包含(空的)JScrollPane,即输入字段和按钮应该消失。但是,它们会一直保留到调整组件大小为止... public
我有一个超过 400 行的列表。每行看起来都类似于:example-example123 我想删除“-”之后的所有内容,这样我只剩下开头部分:example123任何帮助将不胜感激。 最佳答案 像这样
我有两个 for 循环来从列表中删除项目。我正在为这些循环寻找等效的 LINQ 语句 for (Int32 i = points.Count - 1; i >= 0; i--) { for (
我有一个页面,将 View 模型绑定(bind)到 jQuery UI 对话框内的 HTML 表。 当用户关闭对话框时,我想删除 viewmodel observableArray 中绑定(bind)
myGenericList.RemoveAll(x => (x.StudentName == "bad student")); 效果很好,但绑定(bind)列表没有此方法。如何为绑定(bind)列表创
我使用数组列表来查找两个字符串(即 str2 和 str3)之间的差异。当我使用下面的代码时,它工作正常并返回预期的输出。但当我更换时 str2 = #19, 6th cross, 7th main
我有一个小问题,数组列表中的元素没有被删除。这是一个数组列表。这是我的代码: package net.lucrecious.armorconstruct.helpers; import java.ut
我有成员类的简单ArrayList: ArrayList mGroupMembers = new ArrayList<>(); ArrayList mFriends = new ArrayList<>
我预计结果如下,但实际上没有。尽管当我尝试使用字符串而不是项目对象时它起作用了。我想知道为什么会这样以及如何编码以获得预期结果。谢谢。 EXPECTED -----------------------
我有以下代码: ActionListener listenerComboVehicle = new ActionListener() { @Override public void a
(我已经根据“removeall where”或“removeall two argument predicate”的关键字做了尽可能多的搜索,但运气不佳,所以这里开始) 问题是我有一个对象列表(Wa
在通过 NHibernate 检索集合时,我遇到了无法使用 .RemoveAll 的问题。 我有一个名为 Order 的实体,我通过 NHibernate 保留它。 Order 有很多 OrderIt
我有两个列表,我们称它们为列表 A 和列表 B。这两个列表都包含名称并且没有重复项(它们是唯一值)。列表 B 中的每个名称都可以在列表 A 中找到。我想找出列表 B 中缺少哪些名称,以便将这些缺少的名
我是一名优秀的程序员,十分优秀!