【发布时间】:2008-11-13 09:35:23
【问题描述】:
我想做 sort(1) 的逆操作:在 Perl 中将 stdin 的每一行随机化为 stdout。
【问题讨论】:
-
GNU coreutils' shuf [gnu.org/software/coreutils/manual/html_node/… 正是这样做的,但在 C 中。
我想做 sort(1) 的逆操作:在 Perl 中将 stdin 的每一行随机化为 stdout。
【问题讨论】:
我敢打赌,真正的 Perl 黑客会撕毁它,但它仍然存在。
use strict;
use warnings;
use List::Util 'shuffle';
my @lines = ();
my $bufsize = 512;
while(<STDIN>) {
push @lines, $_;
if (@lines == $bufsize) {
print shuffle(@lines);
undef @lines;
}
}
print shuffle(@lines);
这个和其他解决方案的区别:
【讨论】:
这个 perl sn-p 可以解决问题:
#! /usr/bin/perl
# randomize cat
# fisher_yates_shuffle code copied from Perl Cookbook
# (By Tom Christiansen & Nathan Torkington; ISBN 1-56592-243-3)
use strict;
my @lines = <>;
fisher_yates_shuffle( \@lines ); # permutes @array in place
foreach my $line (@lines) {
print $line;
}
# fisher_yates_shuffle( \@array ) : generate a random permutation
# of @array in place
sub fisher_yates_shuffle {
my $array = shift;
my $i;
for ($i = @$array; --$i; ) {
my $j = int rand ($i+1);
next if $i == $j;
@$array[$i,$j] = @$array[$j,$i];
}
}
__END__
【讨论】:
use List::Util 'shuffle';
print shuffle <>
或者如果你担心最后几行缺少\n,
chomp(my @lines = <>);
print "$_\n" for shuffle @lines;
【讨论】: