假设我有一个ArrayList
ArrayList<MyClass> myList;
我想调用toArray,是否有使用性能的理由
MyClass[] arr = myList.toArray(new MyClass[myList.size()]);
过度
MyClass[] arr = myList.toArray(new MyClass[0]);
?
我更喜欢第二种样式,因为它不太冗长,并且我假设编译器将确保不会真正创建空数组,但是我一直在想那是否是真的。
当然,在99%的情况下,它不会以一种方式或其他方式产生任何差异,但我想在我的常规代码和优化的内部循环之间保持一致的样式…
违反直觉的是,在Hotspot 8上最快的版本是:
MyClass[] arr = myList.toArray(new MyClass[0]);
我已经使用jmh运行了一个微型基准测试,结果和代码如下所示,显示具有空数组的版本始终优于具有预定大小的数组的版本。请注意,如果你可以重复使用正确大小的现有数组,则结果可能会有所不同。
基准测试结果(分数以微秒为单位,较小=较好):
Benchmark (n) Mode Samples Score Error Units
c.a.p.SO29378922.preSize 1 avgt 30 0.025 ▒ 0.001 us/op
c.a.p.SO29378922.preSize 100 avgt 30 0.155 ▒ 0.004 us/op
c.a.p.SO29378922.preSize 1000 avgt 30 1.512 ▒ 0.031 us/op
c.a.p.SO29378922.preSize 5000 avgt 30 6.884 ▒ 0.130 us/op
c.a.p.SO29378922.preSize 10000 avgt 30 13.147 ▒ 0.199 us/op
c.a.p.SO29378922.preSize 100000 avgt 30 159.977 ▒ 5.292 us/op
c.a.p.SO29378922.resize 1 avgt 30 0.019 ▒ 0.000 us/op
c.a.p.SO29378922.resize 100 avgt 30 0.133 ▒ 0.003 us/op
c.a.p.SO29378922.resize 1000 avgt 30 1.075 ▒ 0.022 us/op
c.a.p.SO29378922.resize 5000 avgt 30 5.318 ▒ 0.121 us/op
c.a.p.SO29378922.resize 10000 avgt 30 10.652 ▒ 0.227 us/op
c.a.p.SO29378922.resize 100000 avgt 30 139.692 ▒ 8.957 us/op
供参考,代码:
@State(Scope.Thread)
@BenchmarkMode(Mode.AverageTime)
public class SO29378922 {
@Param({"1", "100", "1000", "5000", "10000", "100000"}) int n;
private final List<Integer> list = new ArrayList<>();
@Setup public void populateList() {
for (int i = 0; i < n; i++) list.add(0);
}
@Benchmark public Integer[] preSize() {
return list.toArray(new Integer[n]);
}
@Benchmark public Integer[] resize() {
return list.toArray(new Integer[0]);
}
}
你可以在博文Array of the Wisdom of the Ancients中找到类似的结果,完整的分析和讨论。总结一下:JVM和JIT编译器包含一些优化程序,使其能够廉价地创建和初始化一个新的正确大小的数组,而如果你自己创建该数组,则不能使用这些优化程序。
为什么在数组中需要参数新字符串[0]?
Creates an array from the key/value pairs in this sequence. Signature ObjectLikeSequence.toArray = function() { /*...*/ } ObjectLikeSequence.toArray = function toArray() { return this.pairs().toArra
Creates an array snapshot of a sequence. Note that for indefinite sequences, this method may raise an exception or (worse) cause the environment to hang. Signature Sequence.toArray = function() { /*..
问题内容: 我正在尝试将Set转换为Array。 而且效果很好。但是我不了解in 的意义。 我的意思是起初我正在尝试,但是没有用。为什么需要。 问题答案: 如果Set的元素足够大,则该数组将存储在其中。否则,将为此分配一个具有相同运行时类型的新数组。 Object [] toArray() ,返回不能转换为或任何其他类型数组的。 T [] toArray(T [] a) , 返回包含此集合中所有元
问题内容: 我是Node.js和MongoDB的新手,正在尝试组装自己的博客应用程序。我在尝试通过“博客”模型查询具有特定用户名的用户时遇到问题。当我尝试运行时: 我收到一个错误: 我知道全局变量不好,但是我一直在努力使其正常运行。Mongo文档声称返回了一个游标,该游标可以具有调用的方法。我不知道为什么它不起作用。 这是我的架构/模型创建: 这是/ login和/ readblog请求 最终结果
我一直在学习如何使用java编程,但我还没有明确解释的和方法的区别。第二个只是将LinkedList对象中的所有元素作为数组返回,对吗?但是,第一个呢? 编辑: 我的意思是,我从甲骨文中阅读了文档,它说: 返回一个数组,其中包含此列表中所有元素的正确顺序(从第一个元素到最后一个元素);返回数组的运行时类型是指定数组的运行时类型。如果列表适合指定的数组,则返回该数组。否则,将使用指定数组的运行时类型