在 Perl 中读取二进制文件

Posted

技术标签:

【中文标题】在 Perl 中读取二进制文件【英文标题】:Reading binary file in Perl 【发布时间】:2015-06-07 04:25:43 【问题描述】:

我有一个二进制文件,其中一个接一个地存储了一堆数据块。数据块格式如下:

Length [byte]   Content       Description
2               0xFFFF        Data block header
4               Epoch time    seconds since 00:00:00 UTC, January 1, 1970
2               value of N    Length of data following this value
N               Data          Data itself

我尝试使用unpack,但是由于数据长度不固定,所以这是错误的。

我需要编写一个子程序,它将读取和解析数据块(每次调用子程序时一个数据块)直到文件结束。

文件正在使用big-endian。

这是我到目前为止所尝试的:

use strict;
use warnings;

my $filename;

if (! $ARGV[0])

    die "Input filename is required";


sub setFile

    $filename = $_[0];


my $inFile = $ARGV[0];

setFile($inFile);

open INFILE, $filename or die "\nUnable to open input file";

binmode INFILE;

my $nbytes;

while (<INFILE>) 
    my( $header, $timestamp_hex, $datalength_hex ) = unpack 'H4 H8 H4', $_;
    my $timestamp = hex($timestamp_hex);
    my $datalength = hex($datalength_hex);
    print "$timestamp $datalength\n";

    for (my $i = 0; $i < $datalength; $i++)
    
        my $data = unpack 'H', $_;
        print "$data";
    
    print "\n";


close INFILE
    or die "Error while closing $filename: $!\n";

【问题讨论】:

您是否尝试在binmode 中打开文件,然后尝试读取该文件。 【参考方案1】:

&lt;INFILE&gt; 毫无意义。它会一直读取,直到找到换行符为止。

如果内存中有整个文件,则可以使用以下内容:

my @fields = unpack('( n N n/a* )*', $file);
while (@fields) 
   my ($sig, $ts, $data) = splice(@fields, 0, 3);
   die "Incorrect signature" if $sig != 0xFFFF;
   process_rec($ts, $data);

如果我们要从数据中单独提取标题,我们可以节省内存并添加一些错误检查。

use constant HEADER_FORMAT => 'nNn';
use constant HEADER_LENGTH => length(pack(HEADER_FORMAT, 0, 0, 0));

while (length($file)) 
   last if length($file) < HEADER_LENGTH;
   my ($sig, $ts, $data_len) = unpack(HEADER_FORMAT, substr($file, 0, HEADER_LENGTH, ''));
   die "Incorrect signature" if $sig != 0xFFFF;
   last  if length($file) < $data_len;
   process_rec($ts, substr($file, 0, $data_len, ''));


die "Premature EOF" if length($file);

从文件句柄读取是第二个 sn-p 的扩展。如果内存中没有整个文件,可以使用以下方法:

use constant HEADER_FORMAT => 'nNn';
use constant HEADER_LENGTH => length(pack(HEADER_FORMAT, 0, 0, 0));
use constant BLOCK_SIZE    => 128*1024;

sub make_fill_to = sub 
   my $fh      =  shift;
   my $buf_ref = \shift;
   my $eof = 0;

   return sub 
      my $bytes_needed = $_[1];
      while (!$eof && length($$buf_ref) < $bytes_needed) 
         my $rv = sysread($fh, $$buf_ref, BLOCK_SIZE, length($$buf_ref));
         die $! if !defined($rv);
         $eof = 1 if !$rv;
      

      return !$eof;
   
;

my $buf = '';
my $fill_to = make_fill_to($fh, $buf);
while (1) 
   $fill_to->(HEADER_LENGTH)
      or last LOOP;
   my ($sig, $ts, $data_len) = unpack(HEADER_FORMAT, substr($buf, 0, HEADER_LENGTH, ''));
   die "Incorrect signature" if $sig != 0xFFFF;
   $fill_to->($data_len)
      or last LOOP;
   process_rec($ts, substr($buf, 0, $data_len, ''));


die "Premature EOF" if length($buf);

当使用select管理多个句柄时,必须先读,所以我习惯这样写。如果将上面的内容重构为将读取放在首位,则如下所示:

use constant HEADER_FORMAT => 'nNn';
use constant HEADER_LENGTH => length(pack(HEADER_FORMAT, 0, 0, 0));
use constant BLOCK_SIZE    => 128*1024;

my $buf = '';
my ($got_header, $sig, $ts, $data_len);
while (1) 
   my $rv = sysread($fh, $buf, BLOCK_SIZE, length($buf));
   die $! if !defined($rv);
   last if !$rv;

   while (1) 
      if (!$got_header) 
         last if length($buf) < HEADER_LENGTH;          
         ($sig, $ts, $data_len) = unpack(HEADER_FORMAT, substr($buf, 0, HEADER_LENGTH, ''));
         die "Incorrect signature" if $sig != 0xFFFF;
         $got_header = 1;
      

      last if length($buf) < $data_len;
      process_rec($ts, substr($buf, 0, $data_len, ''));
      $got_header = 0;
   


die "Premature EOF" if $got_buffer || length($buf);

【讨论】:

以上是关于在 Perl 中读取二进制文件的主要内容,如果未能解决你的问题,请参考以下文章

有没有办法在 Perl 中读取 MATLAB .mat 文件?

读取二进制数据并将其转换为十六进制

在 Perl 中解码 3 字节整数

perl程序如何编译成二进制文件并使用

在perl中计算汉明距离

操作指定文件格式的10个Perl CPAN模块