這些循環並不等同,您主要是在抖動bigint包,它本身與for
vs while
無關。
while循環使用符號'$s *= $i
',但for循環使用'$s = $s * $i
'。 這很簡單,足以證明這些不相同。另外,一個循環計數;另一個倒數。這會影響乘數的大小。這是二階效應 - 但不是完全可以忽略的。
[更新:修改爲僅顯示代碼的一個版本,並具有亞秒的時間。有時候可以認爲打印應該從時間計算中排除;這讓事情變得更加混亂,所以我沒有打擾過。我已修復了以前版本中的錯誤:循環4與循環3相同 - 現在不是。我也對輸出格式進行了調整(儘管可以改進次秒處理 - 這是讀者的一項練習),並且有更好的「進度報告」。]
在一臺Mac Mini(雪豹10.6.2)的時序結果:
Count up $s *= $i: 00:00:12.663337
Count up $s = $s * $i: 00:00:20.686111
Count down $s *= $i: 00:00:14.201797
Count down $s = $s * $i: 00:00:23.269874
腳本:
use Time::HiRes qw(gettimeofday);
use strict;
use warnings;
use bigint;
use constant factorial_of => 13000;
sub delta_t
{
my($tag, $t1, $t2) = @_;
my($d) = int($t2 - $t1);
my($f) = ($t2 - $t1) - $d;
my($s) = sprintf("%.6f", $f);
$s =~ s/^0//;
printf "%-25s %02d:%02d:%02d%s\n",
$tag, int($d/3600), int(($d % 3600)/60), int($d % 60), $s;
}
my $t1 = gettimeofday;
{
my $n = factorial_of;
my $s = 1;
for (my $i = 2; $i <= $n; $i++)
{
$s *= $i;
}
print "$s\n: Loop 1\n";
}
my $t2 = gettimeofday;
delta_t('Count up $s *= $i:', $t1, $t2);
{
my $n = factorial_of;
my $s = 1;
for (my $i = 2; $i <= $n; $i++)
{
$s = $s * $i;
}
print "$s\n: Loop 2\n";
}
my $t3 = gettimeofday;
delta_t('Count up $s *= $i:', $t1, $t2);
delta_t('Count up $s = $s * $i:', $t2, $t3);
{
my $n = factorial_of;
my $s = 1;
for (my $i = $n; $i > 1; $i--)
{
$s *= $i;
}
print "$s\n: Loop 3\n";
}
my $t4 = gettimeofday;
delta_t('Count up $s *= $i:', $t1, $t2);
delta_t('Count up $s = $s * $i:', $t2, $t3);
delta_t('Count down $s *= $i:', $t3, $t4);
{
my $n = factorial_of;
my $s = 1;
for (my $i = $n; $i > 1; $i--)
{
$s = $s * $i;
}
print "$s\n: Loop 4\n";
}
my $t5 = gettimeofday;
delta_t('Count up $s *= $i:', $t1, $t2);
delta_t('Count up $s = $s * $i:', $t2, $t3);
delta_t('Count down $s *= $i:', $t3, $t4);
delta_t('Count down $s = $s * $i:', $t4, $t5);
這裏還有一個更緊湊版本上面的代碼擴展爲測試'while'循環以及'for'循環。它也處理大部分時間問題。唯一不理想的是(它)使用了一些全局變量,並且我稍微在代碼中引用了代碼,因此它全部適合一行而不觸發滾動條(反正在我的顯示器上)。顯然,只需要多做一些工作,就可以將測試包裝到一個數組中,以便迭代地完成測試 - 通過對數組中的信息運行計時器函數的數組進行循環。等等...這是一個SMOP - 簡單的編程問題。 (它會打印階乘的MD5哈希,而不是階乘本身,因爲它比較容易比較結果等等。當我重構上面的代碼時,它指出了一些錯誤。是的,MD5是不安全的 - 但我不使用它的安全性,只是被發現的無意更改)壓縮,以避免斷行
use Time::HiRes qw(gettimeofday);
use Digest::MD5 qw(md5_hex);
use strict;
use warnings;
use bigint;
use constant factorial_of => 13000;
my ($s, $i);
my $l1 = sub {my($n) = @_; for ($i = 2; $i <= $n; $i++) { $s *= $i; }};
my $l2 = sub {my($n) = @_; for ($i = 2; $i <= $n; $i++) { $s = $s * $i; }};
my $l3 = sub {my($n) = @_; for ($i = $n; $i > 1; $i--) { $s *= $i; }};
my $l4 = sub {my($n) = @_; for ($i = $n; $i > 1; $i--) { $s = $s * $i; }};
my $l5 = sub {my($n) = @_; $i = 2; while ($i <= $n) { $s *= $i; $i++; }};
my $l6 = sub {my($n) = @_; $i = 2; while ($i <= $n) { $s = $s * $i; $i++; }};
my $l7 = sub {my($n) = @_; $i = $n; while ($i > 1) { $s *= $i; $i--; }};
my $l8 = sub {my($n) = @_; $i = $n; while ($i > 1) { $s = $s * $i; $i--; }};
sub timer
{
my($n, $code, $tag) = @_;
my $t1 = gettimeofday;
$s = 1;
&$code(factorial_of);
my $t2 = gettimeofday;
my $md5 = md5_hex($s);
printf "Loop %d: %-33s %09.6f (%s)\n", $n, $tag, $t2 - $t1, $md5;
}
my $count = 1;
timer($count++, $l1, 'for - Count up $s *= $i:');
timer($count++, $l2, 'for - Count up $s = $s * $i:');
timer($count++, $l3, 'for - Count down $s *= $i:');
timer($count++, $l4, 'for - Count down $s = $s * $i:');
timer($count++, $l5, 'while - Count up $s *= $i:');
timer($count++, $l6, 'while - Count up $s = $s * $i:');
timer($count++, $l7, 'while - Count down $s *= $i:');
timer($count++, $l8, 'while - Count down $s = $s * $i:');
輸出示例(MD5校驗 - 全部價值爲584b3ab832577fd1390970043efc0ec8
):
Loop 1: for - Count up $s *= $i: 12.853630 (584b3ab8...3efc0ec8)
Loop 2: for - Count up $s = $s * $i: 20.854735 (584b3ab8...3efc0ec8)
Loop 3: for - Count down $s *= $i: 14.798155 (584b3ab8...3efc0ec8)
Loop 4: for - Count down $s = $s * $i: 23.699913 (584b3ab8...3efc0ec8)
Loop 5: while - Count up $s *= $i: 12.972428 (584b3ab8...3efc0ec8)
Loop 6: while - Count up $s = $s * $i: 21.192956 (584b3ab8...3efc0ec8)
Loop 7: while - Count down $s *= $i: 14.555620 (584b3ab8...3efc0ec8)
Loop 8: while - Count down $s = $s * $i: 23.790795 (584b3ab8...3efc0ec8)
我。在相應的'for'循環中始終看到'while'循環的小懲罰(< 1%),但我沒有很好的解釋它。
你可能試圖優化錯誤的東西。我想知道你爲什麼認爲這部分語言如此重要? – Ether 2010-03-20 04:48:17
每次運行它們幾次,它們的平均值可能會大致相同 – CaffGeek 2010-03-20 05:03:36
@Chad,實際上我已經測試了幾次代碼。他們花了不同的時間完成同樣的工作。我認爲@Jonathan Leffler對插圖代碼的解釋非常有意義。 – Mike 2010-03-20 05:41:45