12

我有一种情况,我想创建数据结构的签名:

my $signature = ds_to_sig(
  { foo   => 'bar',
    baz   => 'bundy',
    boing => undef,
    number => 1_234_567,
  }
);

目标应该是,如果数据结构发生变化,那么签名也应该如此。

有没有既定的方法来做到这一点?

4

6 回答 6

16

我认为您正在寻找的是哈希函数。我会推荐这样的方法:

use Storable;
$Storable::canonical = 1;
sub ds_to_sig {
    my $structure = shift;
    return hash(freeze $structure);
}

函数哈希可以是任何哈希函数,例如来自Digest::MD5的函数 md5

于 2008-10-20T14:05:46.337 回答
10

最好的方法是使用像Storable这样的深度结构序列化系统。具有相同数据的两个结构将产生相同的可存储输出 blob,因此可以对它们进行比较。

#!/usr/bin/perl

use strict;
use warnings;

use Storable ('freeze');

$Storable::canonical = 1;

my $one = { foo => 42, bar => [ 1, 2, 3 ] };
my $two = { foo => 42, bar => [ 1, 2, 3 ] };

my $one_s = freeze $one;
my $two_s = freeze $two;

print "match\n" if $one_s eq $two_s;

...并证明相反的:

$one = [ 4, 5, 6 ];
$one_s = freeze $one;

print "no match" if $one_s ne $two_s;
于 2008-10-20T14:15:53.820 回答
7

使用Storable ::nstore 将其转换为二进制表示,然后计算校验和(例如使用 Digest 模块)。

这两个模块都是核心模块。

于 2008-10-20T14:13:30.673 回答
5
Digest::MD5->new->add(
  Data::Dumper->new([$structure])
   ->Purity(0)
   ->Terse(1)
   ->Indent(0)
   ->Useqq(1)
   ->Sortkeys(1)
   ->Dump()
)->b64digest();
于 2008-10-20T14:26:37.480 回答
0

我认为您正在寻找的词是“散列”

基本上,您将数据结构放置在一个从中生成相当独特值的函数中。该值将是您的签名。

于 2008-10-20T14:03:34.627 回答
-5

你不能使用对象而不是结构吗?这样您就可以查看对象是否是类型的实例,而无需比较哈希等。

于 2008-10-20T14:07:07.287 回答