PerlによるXMLからCSVへの変換について
Perl初心者です。
下記のようなXMLファイル(test.xml)を、CSVに変換したいと考えています。
<MODESMESSAGE>
<DATETIME>20121129045500</DATETIME>
<MODES>8515C4</MODES>
<CALLSIGN>JAL1406</CALLSIGN>
<ALTITUDE>03075</ALTITUDE>
<GROUNDSPEED>177</GROUNDSPEED>
<TRACK>330</TRACK>
<VRATE>-832</VRATE>
<AIRSPEED>000</AIRSPEED>
<LATITUDE>35.4003</LATITUDE>
<LONGITUDE>139.8822</LONGITUDE>
<SQUAWK>3341</SQUAWK>
</MODESMESSAGE>
<MODESMESSAGE>
<DATETIME>20121129045500</DATETIME>
<MODES>86CF11</MODES>
<ALTITUDE>20100</ALTITUDE>
<VRATE>0</VRATE>
<AIRSPEED>000</AIRSPEED>
<LATITUDE>0</LATITUDE>
<LONGITUDE>0</LONGITUDE>
</MODESMESSAGE>
データによって、要素が欠けている場合もありますが、最大で11個の要素が並んでいます。
これをCSVに変換するため、書籍を参考にしながら以下のスクリプトxml2csv.plを作りました。
使用している環境はLinuxです。
use encoding "utf8", STDOUT=>"Shift_JIS";
use XML::Parser;
$parser = new XML::Parser( Handlers =>
{Start=>\&element_start,
Char => \&cdata,
End => \&element_end});
$element_name = '';
%records = {};
eval ( "\$parser->parsefile( \@ARGV[0] );");
if ($@) {
chomp($@);
$@ =~ s/^\n//;
$@ =~ s/ at (?:[A-Z]:)?\/.*//;
print @ARGV[0] . " :Error $@ \n";
}
sub element_start {
my ($expat, $name, %attributes) = @_;
$element_name = $name;
$records = {} if ( $name eq 'MODESMESSAGE' );
}
sub element_end {
my ($expat, $name) = @_;
if ( $name eq 'MODESMESSAGE' ) {
print $records{'DATETIME'} . "," . $records{'MODES'} . "," . $records{'CALLSIGN'} . "," . $records{'ALTITUDE'} . "," . $records{'GROUNDSPEED'} . "," . $records{'TRACK'} . "," . $records{'VRATE'} . "," . $records{'AIRSPEED'} . "," . $records{'LATITUDE'} . "," . $records{'LONGITUDE'} . "," . $records{'SQUAWK'} . "\n";
}
$element_name = '';
}
sub cdata {
my ($expat, $context) = @_;
$records{$element_name} = $context if ($element_name);
}
実行すると、
> perl xml2csv.pl test.xml
20121129045500,8515C4,JAL1406,03075,177,330,-832,000,35.4003,139.8822,3341
test.xml :Error junk after document element at line 14, column 0, byte 345
>
上記の通り1つ目のデータについてはCSVに変換できているようなのですが、2つ目以降が変換されずエラーとなってしまいます。
お気づきの点がありましたら教えていただけないでしょうか。
よろしくお願いします。
お礼
さっそく素晴らしい回答をくださいまして、ありがとうございました。 本のページ上でも、ちゃんと my @barney = above_average(100, 1..10); print "\@barney = @barney\n"; となっていました。 @を$と一文字書き間違えてただけで、コンパイルエラーにもならずに全然違った結果が得られるんだということを、身に染みて理解できました。 たいへん勉強になりました。ありがとうございました。