ruby – 为什么array.min这么慢?

我注意到array.min似乎很慢,所以我做了这个测试,反对我自己的天真的实现:
require 'benchmark'
array = (1..100000).to_a.shuffle

Benchmark.bmbm(5) do |x|
  x.report("lib:") { 99.times { min = array.min } }
  x.report("own:") { 99.times { min = array[0]; array.each { |n| min = n if n < min } } }
end

结果:

Rehearsal -----------------------------------------
lib:    1.531000   0.000000   1.531000 (  1.538159)
own:    1.094000   0.016000   1.110000 (  1.102130)
-------------------------------- total: 2.641000sec

            user     system      total        real
lib:    1.500000   0.000000   1.500000 (  1.515249)
own:    1.125000   0.000000   1.125000 (  1.145894)

我感到震惊.我自己的实现如何通过每个内部运行一个块?并打败了这么多?

我不知怎么错了?还是这样正常?我很困惑.

我的Ruby版本,在Windows 8.1 Pro上运行:

C:\>ruby --version
ruby 2.2.3p173 (2015-08-18 revision 51636) [i386-mingw32]

解决方法

看看 Enumerable#min的实现.它可能会最终使用每个元素来循环遍历元素,然后在它之前进行一些额外的检查,看它是否需要返回多个元素,或者如果需要通过传递块比较元素.在您的情况下,元素将通过 min_i功能进行比较,我怀疑速度差异来自哪个 – 该功能将比简单比较两个数字慢.

数组没有额外的优化,所有枚举都以相同的方式遍历.

相关文章

以下代码导致我的问题: class Foo def initialize(n=0) @n = n end attr_accessor :n d...
这是我的spec文件,当为上下文添加测试“而不是可单独更新用户余额”时,我得到以下错误. require 's...
我有一个拦截器:DevelopmentMailInterceptor和一个启动拦截器的inititializer setup_mail.rb. 但我想将...
例如,如果我有YAML文件 en: questions: new: 'New Question' other: recent: ...
我听说在RSpec中避免它,let,let !,指定,之前和主题是最佳做法. 关于让,让!之前,如果不使用这些,我该如...
我在Rails中使用MongoDB和mongo_mapper gem,项目足够大.有什么办法可以将数据从Mongoid迁移到 Postgres...