perl:随机播放值排序的哈希值?

发布于 2024-12-18 17:10:54 字数 422 浏览 1 评论 0原文

首先抱歉我的英语 - 我希望你能理解我。

有一个散列:

$hash{a} = 1;
$hash{b} = 3;
$hash{c} = 3;
$hash{d} = 2;
$hash{e} = 1;
$hash{f} = 1;

我想按值(而不是键)对它进行排序,所以我有:

for my $key ( sort { $hash{ $a } <=> $hash{ $b } } keys %hash  ) { ... }

首先我得到所有值为 1 的键,然后值为 2 等......太棒了。

但如果散列没有改变,键的顺序(按值排序)总是相同的。

问题:如何打乱排序结果,以便每次运行“for”循环时,我都会得到不同顺序的键值 1、值 2 等?

At first sorry for my english - i hope you will understand me.

There is a hash:

$hash{a} = 1;
$hash{b} = 3;
$hash{c} = 3;
$hash{d} = 2;
$hash{e} = 1;
$hash{f} = 1;

I want to sort it by values (not keys) so I have:

for my $key ( sort { $hash{ $a } <=> $hash{ $b } } keys %hash  ) { ... }

And at first I get all the keys with value 1, then with value 2, etc... Great.

But if hash is not changing, the order of keys (in this sort-by-value) is always the same.

Question: How can I shuffle sort-results, so every time I run 'for' loop, I get different order of keys with value 1, value 2, etc. ?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(5

哆啦不做梦 2024-12-25 17:10:54

不太确定我很了解您的需求,但这可以吗:

use List::Util qw(shuffle);

my %hash;
$hash{a} = 1;
$hash{b} = 3;
$hash{c} = 3;
$hash{d} = 2;
$hash{e} = 1;
$hash{f} = 1;

for my $key (sort { $hash{ $a } <=> $hash{ $b } } shuffle( keys %hash  )) {
    say "hash{$key} = $hash{$key}"
}

Not quite sure I well understand your needs, but is this ok:

use List::Util qw(shuffle);

my %hash;
$hash{a} = 1;
$hash{b} = 3;
$hash{c} = 3;
$hash{d} = 2;
$hash{e} = 1;
$hash{f} = 1;

for my $key (sort { $hash{ $a } <=> $hash{ $b } } shuffle( keys %hash  )) {
    say "hash{$key} = $hash{$key}"
}
私野 2024-12-25 17:10:54

您可以简单地添加另一个排序级别,当常规排序方法无法区分两个值时将使用该排序级别。例如:

sort { METHOD_1 || METHOD_2 || ... METHOD_N } LIST

例如:

sub regular_sort {
    my $hash = shift;
    for (sort { $hash->{$a} <=> $hash->{$b} } keys %$hash) {
        print "$_ ";
    };
}
sub random_sort {
    my $hash = shift;
    my %rand = map { $_ => rand } keys %hash;
    for (sort { $hash->{$a} <=> $hash->{$b} ||
        $rand{$a} <=> $rand{$b} } keys %$hash ) {
        print "$_ ";
    };
}

You can simply add another level of sorting, which will be used when the regular sorting method cannot distinguish between two values. E.g.:

sort { METHOD_1 || METHOD_2 || ... METHOD_N } LIST

For example:

sub regular_sort {
    my $hash = shift;
    for (sort { $hash->{$a} <=> $hash->{$b} } keys %$hash) {
        print "$_ ";
    };
}
sub random_sort {
    my $hash = shift;
    my %rand = map { $_ => rand } keys %hash;
    for (sort { $hash->{$a} <=> $hash->{$b} ||
        $rand{$a} <=> $rand{$b} } keys %$hash ) {
        print "$_ ";
    };
}
御弟哥哥 2024-12-25 17:10:54

要按值对键进行排序,并对具有相同值的键进行随机排序,我看到两种解决方案:

use List::Util qw( shuffle );
use sort 'stable';
my @keys =
   sort { $hash{$a} <=> $hash{$b} }
   shuffle keys %hash;

或者

my @keys =
   map $_->[0],
   sort { $a->[1] <=> $b->[1] || $a->[2] <=> $b->[2] }
   map [ $_, $hash{$_}, rand ],
   keys %hash;

需要 use sort 'stable'; 来防止 sort 损坏shuffle 返回的列表的随机性。


上面对 Schwartzian 变换 的使用并不是优化的尝试。我见过人们在比较函数本身中使用 rand 来尝试实现上述结果,但这样做存在错误,原因有两个。

当使用诸如此类的“行为不当”比较时,结果会被记录为未定义,因此sort允许返回垃圾、重复元素、缺失元素等

sort 不会返回垃圾,它不会是公平的排序。将对结果进行权衡。

To sort the keys by value, with random ordering of keys with identical values, I see two solutions:

use List::Util qw( shuffle );
use sort 'stable';
my @keys =
   sort { $hash{$a} <=> $hash{$b} }
   shuffle keys %hash;

or

my @keys =
   map $_->[0],
   sort { $a->[1] <=> $b->[1] || $a->[2] <=> $b->[2] }
   map [ $_, $hash{$_}, rand ],
   keys %hash;

The use sort 'stable'; is required to prevent sort from corrupting the randomness of the list returned by shuffle.


The above's use of the Schwartzian Transform is not an attempt at optimisation. I've seen people use rand in the compare function itself to try to achieve the above result, but doing so is buggy for two reasons.

When using "misbehaving" comparisons such as that, the results are documented as being undefined, so sort is allowed to return garbage, repeated elements, missing elements, etc.

Even if sort doesn't return garbage, it won't be a fair sort. The result will be weighed.

零度° 2024-12-25 17:10:54

您可以有两个用于升序和降序的函数,并相应地使用它们,例如

sub hasAscending {
   $hash{$a} <=> $hash{$b};
}

sub hashDescending {
   $hash{$b} <=> $hash{$a};
}

foreach $key (sort hashAscending (keys(%hash))) {
   print "\t$hash{$key} \t\t $key\n";
}

foreach $key (sort hashDescending (keys(%hash))) {
   print "\t$hash{$key} \t\t $key\n";
}

You can have two functions for ascending and decending order and use them accordingly like

sub hasAscending {
   $hash{$a} <=> $hash{$b};
}

sub hashDescending {
   $hash{$b} <=> $hash{$a};
}

foreach $key (sort hashAscending (keys(%hash))) {
   print "\t$hash{$key} \t\t $key\n";
}

foreach $key (sort hashDescending (keys(%hash))) {
   print "\t$hash{$key} \t\t $key\n";
}
心病无药医 2024-12-25 17:10:54

看来您想随机化按键循环。

Perl 不按顺序或排序顺序存储,但这对您来说似乎不够随机,因此您可能需要创建一个键数组并循环遍历它。

首先,用键填充数组,然后使用随机数算法 (1..$#length_of_array) 将数组中该位置的键推送到 array_of_keys。


如果您尝试随机化按值排序哈希的键,那就有点不同了。

查看键盘

my %hash = (a=>1, b=>3, c=>3, d=>2, e=>1, f=>1);
my %hash_by_val;

for my $key ( sort { $hash{$a} <=> $hash{$b} } keys %hash ) { 
   push @{ $hash_by_val{$hash{$key}} }, $key;
}


for my $key (sort keys %hash_by_val){
   my @arr        = @{$hash_by_val{$key}};
   my $arr_ubound = $#arr;

   for (0..$arr_ubound){
      my $randnum = int(rand($arr_ubound));
      my $val     = splice(@arr,$randnum,1);
      $arr_ubound--;
      print "$key : $val\n";                    # notice: output varies b/t runs
   }
}

It seems like you want to randomize looping through the keys.

Perl, does not store in sequential or sorted order, but this doesn't seem to be random enough for you, so you may want to create an array of keys and loop through that instead.

First, populate an array with keys, then use a random number algorithm (1..$#length_of_array) to push the key at that position in the array, to the array_of_keys.


If you're trying to randomize the keys of the sorted-by-value hash, that's a little different.

See Codepad

my %hash = (a=>1, b=>3, c=>3, d=>2, e=>1, f=>1);
my %hash_by_val;

for my $key ( sort { $hash{$a} <=> $hash{$b} } keys %hash ) { 
   push @{ $hash_by_val{$hash{$key}} }, $key;
}


for my $key (sort keys %hash_by_val){
   my @arr        = @{$hash_by_val{$key}};
   my $arr_ubound = $#arr;

   for (0..$arr_ubound){
      my $randnum = int(rand($arr_ubound));
      my $val     = splice(@arr,$randnum,1);
      $arr_ubound--;
      print "$key : $val\n";                    # notice: output varies b/t runs
   }
}
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文