Если вы не можете гарантировать, что первая координата всегда будет уникальной, идея пары лучше представить в виде отдельного массива из двух элементов. Вы также можете гораздо проще распространить ту же идею на кортежи более высоких измерений.
#!/usr/bin/perl
use strict; use warnings;
use Data::Dumper;
my @tuples = ([qw(A P)], [qw(B Q)], [qw(C R)]);
my $re_tmpl = '^%s.*%s=(.)';
my @re = map qr/$_/, map sprintf($re_tmpl, @$_), @tuples;
while (my $line = <DATA>) {
last unless $line =~ /\S/;
my ($value) = map { $line =~ $_ } @re;
print $value, "\n";
}
__DATA__
A P=1 Q=2 R=3
B P=8 Q=2 R=7
C Q=2 P=1 R=3
Но, используя ваш метод и метод выше, вы выполняете больше операций сопоставления, чем необходимо (три на строку, а не одна). Это делает @ ответ Евгения более эффективным.
Более общим решением является использование:
#!/usr/bin/perl
use strict; use warnings;
my @tuples = ([qw(A P)], [qw(B Q)], [qw(C R)]);
my $re_tmpl = '^%s.*%s=(.)';
my %re;
@re{ map $_->[0], @tuples } = map qr/$_/,
map sprintf($re_tmpl, @$_),
@tuples;
while (my $line = <DATA>) {
last unless $line =~ /\S/;
my ($value) = $line =~ $re{substr $line, 0, 1};
print $value, "\n";
}
__DATA__
A P=1 Q=2 R=3
B P=8 Q=2 R=7
C Q=2 P=1 R=3
Приятно то, что вы можете адаптировать его для кортежей с размерами, превышающими два.
Кроме того, теперь, когда вы выбираете шаблон на основе первого символа строки, сами шаблоны становятся проще:
#!/usr/bin/perl
use strict; use warnings;
my @tuples = ([qw(A P)], [qw(B Q)], [qw(C R)]);
my $re_tmpl = '%s=(.)';
my %re;
@re{ map $_->[0], @tuples } = map qr/$_/,
map sprintf($re_tmpl, $_->[1]),
@tuples;
while (my $line = <DATA>) {
last unless $line =~ /\S/;
my ($value) = $line =~ $re{substr $line, 0, 1};
print $value, "\n";
}
__DATA__
A P=1 Q=2 R=3
B P=8 Q=2 R=7
C Q=2 P=1 R=3
Более простая альтернатива (которая предполагает захват всех x=y
):
#!/usr/bin/perl
use strict; use warnings;
my %pairs = qw(A P B Q C R);
my $re = qr/([A-Z])=([0-9])/;
while (my $line = <DATA>) {
last unless $line =~ /\S/;
my $type = substr $line, 0, 1;
my $value = { $line =~ /$re/g }->{ $pairs{$type} };
print "$value\n";
}
__DATA__
A P=1 Q=2 R=3
B P=8 Q=2 R=7
C Q=2 P=1 R=3
Последний также облегчает получение нескольких значений из строки:
#!/usr/bin/perl
use strict; use warnings;
my %tuples = (A => [qw(P Q)], B => [qw(Q R)], C => [qw(P R)]);
my $re = qr/([A-Z])=([0-9])/;
while (my $line = <DATA>) {
last unless $line =~ /\S/;
my $type = substr $line, 0, 1;
my @values = @{ { $line =~ /$re/g } }{ @{ $tuples{$type} } };
print "@values\n";
}
__DATA__
A P=1 Q=2 R=3
B P=8 Q=2 R=7
C Q=2 P=1 R=3