4

我正在尝试将变量$num转换为其反向字节顺序并将其打印出来:

my $num = 0x5514ddb7;
my $s = pack('I!',$num);
print "$s\n";

它打印为一些不可打印的字符,在十六进制编辑器中看起来不错,但我怎样才能让它在控制台上可读?我已经尝试过了:

print sprintf("%#x\n",$s);

这使得 perl 抱怨一个非数字参数,所以我认为pack返回一个字符串。任何想法如何0xb7dd1455在控制台上打印出来,基于$num

4

6 回答 6

7

您需要使用unpack

my $num = 0x5514ddb7;
my $s = pack('I<!',$num);
my $x = unpack('I>!',$s);
printf("%#x\n",$x);

Michael Carman的评论:请注意字节顺序修饰符 (<>) 需要 Perl v5.10+。对于旧版本的 Perl,您必须改用NV 模板

于 2010-06-03T15:23:34.610 回答
2
my $num=0x5514ddb7;
my $i = unpack('N', pack('V',$num));
printf("0x%x\n", $i);

但是你确定要这样做吗?它是 32 位特定的。问题是“你为什么问?” 为了提出比您要解决的问题更好的建议。

于 2010-06-03T15:23:52.397 回答
2

您需要使用CORE::unpack() "H"(大写 H):

H   A hex string (high nybble first).

my $num=0x5514ddb7;
my $s=pack('I!',$num);
print unpack ("H*", "$s") . "\n"; # b7dd1455
于 2010-06-03T15:27:09.593 回答
1

作为一个涉足 perl 的以 C 为主的程序员,我的直觉是使用位运算符进行切片和切块:

$n2 = ($num & 0xff) <<24
    | ($num & 0xff00) <<8
    | ($num & 0xff0000) >>8
    | ($num & 0xff000000) >>24 ;

但是,这是 perl,毫无疑问有很多方法可以达到相同的结果,并且可能有一个很好的 perl 习惯用法。

于 2010-06-03T15:24:16.013 回答
0

我想我会为我们这些需要检查格式化二进制文件的人插话。当规范以非英特尔字节顺序(如 MySQL .frm 规范)显示十六进制值时,这是我使用的。您可以通过增加 sysread 大小来指定更大的块。打包/解包长度必须是 sysread 值的 2 倍。偏移量 += [设置为 sysread 值]。

#!/usr/bin/perl
BEGIN {
    use Getopt::Long;

    $reverse=0;
    GetOptions ("reverse" => \$reverse);

}

my $f=STDIN;
if (int(@ARGV))
{
    open(IN, $ARGV[0]) or die "Failed to open $ARGV[0] for reading.\n$!";
    $f=IN;
}

my $count=1;
my $offset=0;
my $after_nl=1;
while (sysread($f,$buf,2))
{
    my $hex = unpack('H4', $buf);
    $hex=join("",(split(//,$hex))[2,3,0,1]) unless $reverse;
    if (($count % 8) == 0)
    {
        printf "%s\n", $hex;
        $after_nl=1;
        $offset += 2;
    }
    else
    {
        printf "%08x ", $offset if $after_nl;
        printf "%s ", $hex;
        $offset += 2;
        $after_nl=0;
    }

    $count++;
}
于 2013-08-22T22:59:21.473 回答
0
$num = 0x5514ddb7;
print '0x', sprintf( '%08x', $num ) =~ /../g;
于 2013-08-22T23:08:04.440 回答