我遇到了令人沮丧的任务,即尝试对我们刚刚购买的皇家邮政PAF文件做一些事情,我现在已经将所有相关表格加载到我们的SQL服务器中,并根据相关ID将它们链接在一起。我现在需要做的是根据多个规则将适当的地址放在一起,例如:如果仅提供建筑物名称或编号,或者在给定特定邮政编码时提供了邮政信箱。我希望,因为这似乎是一个广泛使用的文件,有人可能已经完成了一些这方面的工作,任何人都有任何想法,因为他们在本周末想要它。
希望有人可以提供帮助,P
答案 0 :(得分:0)
感到沮丧 - 这是一项重大工作 - 很多规则都有很多例外。 我认为你已经看过PAF程序员指南中的所有规则了吗? http://www.royalmail.com/sites/default/files/docs/pdf/programmers_guide_edition_7_v5.pdf 如果你想在周末之前完成它,你最好得到“压缩标准”版本,这实际上是你所追求的扩展地址,但当然它没有对数据库进行规范化。 我很久以前就用关系格式构建了地址。我会看看能否找到代码。
发现了一些丑陋的8岁的Perl - 谨慎使用
while ( <PAF> ) # Reading from mysql view - you can work this out from the fields populated by the split /\t/
{
chomp;
$incount++;
print STDERR "Processed ..... $incount\n" if $incount%1000000 == 0;
my ( $postcode , $aKey , $lkey , $skey ,
$sekey , $dskey , $dsekey , $bnumber ,
$bkey , $sbkey , $households , $oKey ,
$fmtPcType , $concat , $dps , $fmtsUser ,
$poBox , $version , $active ,
$fmtCompany , $fmtDepartment , $rawSubLine , $rawBldLine ,
$DepSteet , $fmtDepSteetEnd , $Steet , $fmtSteetEnd ,
$DepLocality , $Locality , $PostTown )
= split /\t/ , $_ , -1 ;
my $fmtaKey = sprintf("%08s",$aKey);
my $fmtoKey = sprintf("%08s",$oKey);
$fmtoKey = "" if $fmtoKey eq "00000000" ;
$fmtoKey = $fmtaKey if $fmtPcType eq "L";
$fmtDepLocality = removePunch($DepLocality);
$fmtLocality = removePunch($Locality,"dq");
$fmtDepSteet = removePunch($DepSteet);
$fmtSteet = removePunch($Steet,"dh");
$rawSubLine = removePunch($rawSubLine,"d");
$rawBldLine = removePunch($rawBldLine,"d");
my $fmtPostTown = removePunch($PostTown);
my ($opc,$ipc) = unpack("A4A3" ,$postcode);
$opc =~ s/\s$//;
my $fmtPoBox = "PO BOX $poBox" if (length($poBox) > 0);
my ($fmtSubBuilding , $fmtBuilding ) = ( "","") ;
# ##############################################################################
# Format rules are based on presence of builing name , subbuilding and number
# See PAF Digest for details
# ##############################################################################
my ($subBuildingFlag,$buildingFlag,$buildingNumberFlag) = ("N","N","N") ;
$buildingNumberFlag ="Y" if $bnumber ne "0" ;
$buildingFlag ="Y" if length($rawBldLine) > 0;
$subBuildingFlag ="Y" if length($rawSubLine) > 0;
my $formatKey="${subBuildingFlag}${buildingFlag}${buildingNumberFlag}";
my $ruleid ;
# ----------------------------------------------------------------------------
# Formmating Rule 1 (Org name only)
# ----------------------------------------------------------------------------
if ( $formatKey eq "NNN" )
{
$ruleid="1";
}
# ----------------------------------------------------------------------------
# Formmating Rule 2 (Building number only)
# ----------------------------------------------------------------------------
elsif ( $formatKey eq "NNY" )
{
$ruleid="2";
InsertNumber ($bnumber) ;
}
# ----------------------------------------------------------------------------
# Formmating Rule 3 (Building Name only)
# ----------------------------------------------------------------------------
elsif ($formatKey eq "NYN")
{
$ruleid="3";
my $N="";
($fmtBuilding,$N) = split /\|/ , F1Check ($rawBldLine) , -1 ;
InsertNumber ($N) ;
}
# ----------------------------------------------------------------------------
# Formmating Rule 4 (Building Name and building Number)
# ----------------------------------------------------------------------------
elsif ($formatKey eq "NYY")
{
$ruleid="4";
$fmtBuilding = "$rawBldLine";
InsertNumber ($bnumber) ;
}
# ----------------------------------------------------------------------------
# Formmating Rule 5 (SubBuilding Name and building Number)
# ----------------------------------------------------------------------------
elsif ($formatKey eq "YNY")
{
$ruleid="5";
if ($concat eq "Y")
{
my $numSub = "$bnumber $rawSubLine";
InsertNumber ($numSub) ;
}
else
{
$fmtSubBuilding = "$rawSubLine";
InsertNumber ($bnumber) ;
}
}
# ----------------------------------------------------------------------------
# Formmating Rule 6 (SubBuilding Name and building name)
# ----------------------------------------------------------------------------
elsif ($formatKey eq "YYN" )
{
$ruleid="6";
($fmtSubBuilding,$N1) = split /\|/ , F1Check ($rawSubLine) , -1 ;
($fmtBuilding,$N2) = split /\|/ , F1Check ($rawBldLine) , -1 ;
if ( $fmtSubBuilding eq "" )
{
if ( $N2 =~ /^REAR OF/ )
{
$fmtSubBuilding .= "$N1 $N2" ;
$N1 = "" ;
$N2 = "" ;
}
else
{
$fmtSubBuilding .= "$N1 " . $fmtBuilding;
$fmtBuilding = "";
$N1 = "" ;
}
}
$N2 = $N1 if $N2 eq "";
InsertNumber ($N2) ;
}
# ----------------------------------------------------------------------------
# Formmating Rule 7 (SubBuilding Name ,building name and building number)
# ----------------------------------------------------------------------------
elsif ($formatKey eq "YYY" )
{
$ruleid="7";
($fmtSubBuilding,$N1) = split /\|/ , F1Check ($rawSubLine) , -1 ;
($fmtBuilding,$N2) = split /\|/ , F1Check ($rawBldLine) , -1 ;
if ( $fmtSubBuilding eq "" )
{
$fmtSubBuilding .= "$N1 " . $fmtBuilding;
$fmtBuilding = "";
}
InsertNumber ($bnumber) ;
}
# ##############################################################################
# Format the address
# ##############################################################################
#----------------------------------------------------------------
# subbuilding = building if no subbuilding (why?)
#----------------------------------------------------------------
if (( length($fmtBuilding) > 0) && ($fmtSubBuilding eq "" ))
{
$fmtSubBuilding = $fmtBuilding;
$fmtBuilding = "";
}
#----------------------------------------------------------------
# Get rid of duplicate lines
#----------------------------------------------------------------
if ("${fmtSubBuilding}${fmtBuilding}" eq "${fmtSteet}${fmtSteetEnd}")
{
$fmtSubBuilding="";
$fmtBuilding="";
}
#----------------------------------------------------------------
# Parse out number ranges (including number suffixes)
#----------------------------------------------------------------
my ($fmtStreetName ,$lo_num , $low_suf , $hi_num ,$hi_suf) = split /\|/ , getLoHiNum ("$fmtSteet") , -1 ;
my $dependentTfare = "$fmtDepSteet";
$dependentTfare .= " $fmtDepSteetEnd" if length($fmtDepSteetEnd) > 0;
# ABERDEEN CITY COUNCIL|EDUCATION DEPARTMENT||ST NICHOLAS HOUSE||||BROAD|STREET|||ABERDEEN|ABERDEENSHIRE|AB10|1AG|1A|0|01901355|01901355|L||
my $fmtaddr .= "$postcode|$dps|$fmtaKey|$fmtoKey|$fmtPoBox|$fmtSubBuilding|$fmtBuilding|$fmtDepSteet|$fmtDepSteetEnd|" ;
$fmtaddr .= "$fmtSteet|$fmtSteetEnd|$fmtDepLocality|$fmtLocality|$fmtPostTown|$opc|$ipc|" ;
$fmtaddr .= "$ruleid|$lo_num|$low_suf|$hi_num|$hi_suf|$dependentTfare|$fmtStreetName";
print "$fmtaddr\n";
}
close PAF;
# # ############################################################################
# SUB F1Check (PAF Digest Note:1 Page 42)
#
# Try and extract a number embeded in the (sub)building name
# There are loads of exceptions to the rules in the PAF digest
# hence the horrible regex's
# # ############################################################################
sub F1Check
{
my ($building) = (@_);
my $bnumber = "";
if ($building =~ m/^REAR OF.+\d/ )
{
$bnumber=$building;
$building="" ;
}
elsif (($building =~ m/\d+/) && ($building !~ /UNIT[S]?\s|
FLAT[S]?\s|
^DP\d+[A-Z]?$|
BLOCK[S]?\s|
^OFFICE[S]?\s|
HANG[EA]R\s|
^BUILDING[S]?\s|
^HOUSE\s|
HOLDING\s|
APARTMENT\s|
CHALET\s|
^BUNGALOW\s|
^CARAVAN\s|
^ANNEXE\s|
^PLOT[S]?\s|
^HOME[S]?\s|
^LODGE\s\d|
^PLATFORM\s|
^DOMUS\s|
^BARLOW\s|
^BEALAH\s|
^KIOSK[S]?\s|
^LEVEL[S]?\s|
^VILLA\s|
MEOTA\s|
MAXI[NM]\s|
^AQUEOUS\s|
SUITE[S]?\s|
WING\s|
^CAMPUS\s|
^STUDIO\s|
^COTTAGE\s|
^STALL\s|
^SHOP\s|
^ARCH\s|
^QUAY\s|
^ABOVE\s\d|
^LINK\s|
JETTY\s|
WAREHOUSE\s|
^HOLDING\s|
^PENTHOUSE\s|
^MOORING\s|
^BOTHY\s|
^MAISONETTE\s|
^SITE\s|
^WORKSHOP\s|
^BARN[S]?\s|
STALL\s|
^BOAT\s|
^STAND\s|
^TOWER\s\d|
^YARD\s\d|
^STANCE\s|
^VAN\s|
^BAY\s\d|
^MOBILE HOME\s|
^STABLE\s|
^ROOM\s|
^[A-Z]\d+$
/xo
))
{
if ( $building =~ m/^([A-Z]?\d+[A-Z]{0,2}|
\d+[A-Z]{0,2}[\-\&\ \\\/]{0,1}\d+[A-Z]{0,2}|
[A-Z])$/ox )
{
$bnumber=$building;
$building="" ;
}
elsif ($building =~ m/^([A-Z]?\d*\s*[A-Z\s\.]+)(\d+[A-Z]{0,2}[\-\&\ \\\/]{0,1}\d*[A-Z]{0,2})$/o)
{
if (length($1) > 2)
{
$bnumber=$2;
$building=$1;
$building =~ s/\s+$//g;
}
}
elsif ($building =~ m/^(\D+)(\d+[A-Z]{0,2}[\-\&\ \\\/]\d+[A-Z]{0,2})$/)
{
$bnumber=$2;
$building=$1;
$building =~ s/\s+$//g;
}
}
return "$building|$bnumber"
}
# # ############################################################################
# SUB InsertNumber
#
# Prepend number to first non blank line from thoroughfare onwards
# # ############################################################################
sub InsertNumber
{
my ($N) = (@_) ;
if ( $N =~ /\d/ && $N ne "0")
{
if (length($fmtDepSteet) > 0) {$fmtDepSteet = $N . " " . $fmtDepSteet ;}
elsif (length($fmtSteet) > 0) {$fmtSteet = $N . " " . $fmtSteet;}
elsif (length($fmtDepLocality) > 0) {$fmtDepLocality = $N . " " . $fmtDepLocality ;}
else {$fmtLocality = $N . " " . $fmtLocality;} ;
}
}
# # ############################################################################
# SUB getLoHiNum
#
# Extract number ranges
# # ############################################################################
sub getLoHiNum
{
my ($tfare) = (@_) ;
my ($lonum,$losuf,$hinum,$hisuf) = ( "","","","");
# extract numbers and suffixes
my ($num,$tt) = ($tfare=~ /^(\d+[A-Z]?[\-]?\d*[A-Z]?)\s(.*)$/ );
if (length($num) > 0)
{
$tfare = $tt ;
if ( $num =~ m/[\-]{1}/ )
{
($lonum,$hinum) = ($num=~ /(\d+.*)[\-](\d+.*)/ );
}
else
{
$lonum = $num;
}
if ( $lonum =~ m/[A-Z]$/ )
{
($lonum,$losuf) = ($lonum=~ /(\d+)([A-Z])$/ );
}
if ( $hinum =~ m/[A-Z]$/ )
{
($hinum,$hisuf) = ($hinum=~ /(\d+)([A-Z])$/ );
}
}
return "$tfare|$lonum|$losuf|$hinum|$hisuf" ;
}
# # ############################################################################
# SUB removePunch
#
# Remove punchuation
# # ############################################################################
sub removePunch
{
my ($dirtyWord,$punch) = (@_) ;
$punch = "dhqa" if length($punch) == 0;
$dirtyWord =~ s/\.//g if $punch =~ m/d/ ;
$dirtyWord =~ s/\-/ /g if $punch =~ m/h/ ;
$dirtyWord =~ s/\'//g if $punch =~ m/q/ ;
$dirtyWord =~ s/\@//g if $punch =~ m/a/ ;
return $dirtyWord;
}
答案 1 :(得分:0)
如果其他任何人遇到此问题,如果我正确理解了这个问题,要从数据文件中重新构造地址,则RM PAF文件中的列是相反的顺序。因此,要构建一个简单的地址字符串,从右到左检查列,您只需检查每个值是否为一个值,如果有一个值,则将其连接到带有空格的地址字符串中。如果您查看一些使用每个字段的示例,则可以对空格和逗号进行解析。