温馨提示:本文翻译自stackoverflow.com,查看原文请点击:perl - Multiple filehandles opening the same file
perl filehandle

perl - 多个文件句柄打开同一文件

发布于 2020-03-29 21:33:22

我有grades.tsv文件,其中包含三列,分别显示学生的姓名,科目和成绩:

Liam    Mathematics 5
Liam    History 6
Liam    Geography   8
Liam    English 8
Aria    Mathematics 8
Aria    History 7
Aria    Geography   6
Isabella    Mathematics 9
Isabella    History 4
Isabella    Geography   7
Isabella    English 5
Isabella    Music   8

我想计算每个学生的平均成绩,并将其添加到单独的列中。为此,我使用了两个文件句柄DATA和OUT打开同一文件:

use strict;
use warnings;

# Open file with grades for calculation of average grade for each student
open (DATA,"grades.tsv") or die "Cannot open file\n";

my %grade_sums;
my %num_of_subjects;

# Calculate sum of grades and number of subjects for each student
while( <DATA> ) {

   chomp;
   my ($name, $subject, $grade) = split /\t/;

   $grade_sums{$name} += $grade;
   $num_of_subjects{$name} += 1;
}

close DATA;


# Open file with grades again but this time for a purpose of adding a separate column with average grade and printing a result
open (OUT,"grades.tsv") or die "Cannot open file\n";

while ( <OUT> ) {
   chomp;
   my ($name, $subject, $grade) = split /\t/;

   # Calculate average grade
   my $average_grade = $grade_sums{$name} / $num_of_subjects{$name};
   my $outline = join("\t", $name, $subject, $grade, $average_grade);

   # Print a file content with new column
   print "$outline\n";
}

close OUT;

该代码有效,但是我不确定这是否是完成此任务的正确方法。这是一个好习惯还是应该有更好的方法?

查看更多

提问者
zubenel
被浏览
89
ikegami 2020-01-31 01:45

重新打开文件就可以了。一种选择是寻求文件的开头。

use Fcntl qw( SEEK_SET );

seek(DATA, 0, SEEK_SET);

寻求效率更高,因为它不必检查权限等。它还保证您获得相同的文件(但没有人更改过它)。

另一种选择是将整个文件加载到内存中。那就是我通常要做的。


注意

open(FH, $qfn) or die "Cannot open file\n";

最好写成

open(my $FH, '<', $qfn)
   or die("Can't open file \"$qfn\": $!\n");
  • 三参数open避免了一些问题。
  • 在错误消息中包括错误原因是有益的。
  • 在错误消息中包括路径是有益的。
  • 应该避免使用,DATA因为Perl有时会自动使用该名称创建一个句柄。
  • 应该避免使用全局变量(例如FH)或词法变量(my $FH)。