有没有办法在Perl中检查CUSIP的数字

时间:2015-01-22 20:15:33

标签: algorithm perl financial check-digit

我的工作地点不允许我们安装任何模块,因此该选项对我来说并不适合。因此决定查看此链接http://en.wikipedia.org/wiki/CUSIP并按照其中的伪代码尝试在Perl中对其进行编码。

我想出了以下内容:

sub cusip_check_digit
{
    my $cusip = shift;                    ## Input: an 8-character CUSIP
    my $v = 0;                            ## numeric value of the digit c (below)

    my $sum = 0;
    for (my $i = 0; $i < 8; $i++)
    {
        my $c = substr ($cusip, $i, 1);   ## $c is the ith character of cusip          
        if ($c =~ /\d/)                   ## c is a digit then
        {
            $v = $c;                      ## numeric value of the digit c
        }
        elsif ($c =~ /\w/)
        {
            my $p = ord($c) - 64;         ##  ordinal position of c in the alphabet (A=1, B=2...)
            $v = $p + 9;
        }

        if (0 != $i % 2)                  ## check to see if $i is even (we invert due to Perl starting points)
        {
            $v = $v * 2;
        }

        $sum = $sum + int ($v / 10) + $v % 10;

    }

    $v = (10 - ($sum % 10)) % 10;
    print "v is: $v\n";

    #return (10 - ($sum % 10)) % 10
}

cusip_check_digit('90137F10');   ## should return 3 ** Now works **
cusip_check_digit('68243Q10');   ## should return 6 ** Now works **

不完全确定它为什么不起作用。

1 个答案:

答案 0 :(得分:2)

我认为你的问题就在这一行:

$sum = $sum + $v / 10 + $v % 10;

维基说'div'和'mod'。这意味着整数除法,而这不是它正在做的事情。

将其更改为:

$sum = $sum + int ( $v / 10 ) + $v % 10;

你得到'3'的预期结果。我没有用任何其他值检查它,所以你最好检查一下。

编辑:第二个问题是因为我们从0到7而不是像例子中那样从1到8运行。这意味着'i甚至'测试得到错误的数字。通过反转逻辑很容易解决(测试'奇数'而不是'偶数')。

将该位更改为:

if (0 != $i % 2) {
    $v = $v * 2;
}