验证银行卡号正确性的算法

    public static function luhm($cardno)
    {
        //位数小于16,直接返回失败
        if(strlen($cardno) < 16){
            return false;
        }
        $lastNum = substr($cardno,-1);  //取出最后一位(与luhm进行比较)
        $first15Num = substr($cardno,0,-1);//前15或18位
        $newArr = str_split(strrev($first15Num));//前15或18位反转后转换为数组

        $oddArr = [];   //奇数位*2的积 <9
        $oddArr2 = [];  //奇数位*2的积 >9
        $evenArr = [];  //偶数位数组
        for($j=0;$j<count($newArr);$j++) {
            if (($j + 1) % 2 == 1) {    //奇数位
                if ($newArr[$j] * 2 < 9) {
                    $oddArr[] = $newArr[$j] * 2;
                } else {
                    $oddArr2[] = $newArr[$j] * 2;
                }
            } else {//偶数位
                $evenArr[] = $newArr[$j];
            }
        }
        $oddChild1 = []; //奇数位*2 >9 的分割之后的数组个位数
        $oddChild2 = []; //奇数位*2 >9 的分割之后的数组十位数
        foreach($oddArr2 as $ov){
            $oddChild1[] = $ov % 10;
            $oddChild2[] = floor($ov / 10);
        }

        $sumOdd = array_sum($oddArr);   //奇数位*2 < 9 的数组之和
        $sumEven = array_sum($evenArr); //偶数位数组之和
        $sumOddChild1 = array_sum($oddChild1); //奇数位*2 >9 的分割之后的数组个位数之和
        $sumOddChild2 = array_sum($oddChild2); //奇数位*2 >9 的分割之后的数组十位数之和

        //计算总和
        $sumTotal = ($sumOdd + $sumEven + $sumOddChild1 + $sumOddChild2);

        //计算Luhm值
        $k = ($sumTotal % 10 == 0) ? 10 : ($sumTotal % 10);
        $luhm = 10 - $k;

        if($lastNum == $luhm){
            return true;
        } else {
            return false;
        }
    }
}

猜你喜欢

转载自blog.csdn.net/fish_study_csdn/article/details/80974973