一般验证银行卡有效性用到一种叫做LUHN的算法,简介请参考这篇博客:基于Luhn算法的银行卡卡号的格式校验
注意:
1、LUHN算法只是能校验卡号是否有效,并不能校验卡号和用户名是否一致。
2、如果有部分银行卡不符合LUHN算法的话,也不用奇怪,有这种可能。
下面贴出JS实现的LUHN算法:
1 /** 2 * 检查银行卡号是否符合规则 3 * @param bankno 银行卡号 4 * @returns 5 */ 6 function checkBankNo(bankno) { 7 var bankno = bankno.replace(/s/g, ''); 8 if (bankno == "") { 9 alert("请填写银行卡号"); 10 return false; 11 } 12 if (bankno.length < 16 || bankno.length > 19) { 13 alert("银行卡号长度必须在16到19之间"); 14 return false; 15 } 16 var num = /^d*$/;// 全数字 17 if (!num.exec(bankno)) { 18 alert("银行卡号必须全为数字"); 19 return false; 20 } 21 // 开头两位 22 var strBin = "10,18,30,35,37,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,58,60,62,65,68,69,84,87,88,94,95,98,99"; 23 if (strBin.indexOf(bankno.substring(0, 2)) == -1) { 24 alert("银行卡号开头两位不符合规范"); 25 return false; 26 } 27 // luhn校验 28 if (!luhnCheck(bankno)) { 29 return false; 30 } 31 return true; 32 } 33 34 /** 35 * 银行卡号luhn校验算法 36 * luhn校验规则:16位银行卡号(19位通用): 37 * 1.将未带校验位的 15(或18)位卡号从右依次编号 1 到 15(18),位于奇数位号上的数字乘以 2。 38 * 2.将奇位乘积的个十位全部相加,再加上所有偶数位上的数字。 39 * 3.将加法和加上校验位能被 10 整除。 40 * @param bankno 银行卡号 41 * @returns 42 */ 43 function luhnCheck(bankno) { 44 var lastNum = bankno.substr(bankno.length - 1, 1);// 取出最后一位(与luhn进行比较) 45 var first15Num = bankno.substr(0, bankno.length - 1);// 前15或18位 46 var newArr = new Array(); 47 for (var i = first15Num.length - 1; i > -1; i--) { // 前15或18位倒序存进数组 48 newArr.push(first15Num.substr(i, 1)); 49 } 50 var arrJiShu = new Array(); // 奇数位*2的积 <9 51 var arrJiShu2 = new Array(); // 奇数位*2的积 >9 52 var arrOuShu = new Array(); // 偶数位数组 53 for (var j = 0; j < newArr.length; j++) { 54 if ((j + 1) % 2 == 1) {// 奇数位 55 if (parseInt(newArr[j]) * 2 < 9) { 56 arrJiShu.push(parseInt(newArr[j]) * 2); 57 } else { 58 arrJiShu2.push(parseInt(newArr[j]) * 2); 59 } 60 } else { 61 arrOuShu.push(newArr[j]);// 偶数位 62 } 63 } 64 65 var jishu_child1 = new Array();// 奇数位*2 >9 的分割之后的数组个位数 66 var jishu_child2 = new Array();// 奇数位*2 >9 的分割之后的数组十位数 67 for (var h = 0; h < arrJiShu2.length; h++) { 68 jishu_child1.push(parseInt(arrJiShu2[h]) % 10); 69 jishu_child2.push(parseInt(arrJiShu2[h]) / 10); 70 } 71 var sumJiShu = 0; // 奇数位*2 < 9 的数组之和 72 var sumOuShu = 0; // 偶数位数组之和 73 var sumJiShuChild1 = 0; // 奇数位*2 >9 的分割之后的数组个位数之和 74 var sumJiShuChild2 = 0; // 奇数位*2 >9 的分割之后的数组十位数之和 75 var sumTotal = 0; 76 for (var m = 0; m < arrJiShu.length; m++) { 77 sumJiShu = sumJiShu + parseInt(arrJiShu[m]); 78 } 79 for (var n = 0; n < arrOuShu.length; n++) { 80 sumOuShu = sumOuShu + parseInt(arrOuShu[n]); 81 } 82 for (var p = 0; p < jishu_child1.length; p++) { 83 sumJiShuChild1 = sumJiShuChild1 + parseInt(jishu_child1[p]); 84 sumJiShuChild2 = sumJiShuChild2 + parseInt(jishu_child2[p]); 85 } 86 // 计算总和 87 sumTotal = parseInt(sumJiShu) + parseInt(sumOuShu) 88 + parseInt(sumJiShuChild1) + parseInt(sumJiShuChild2); 89 // 计算luhn值 90 var k = parseInt(sumTotal) % 10 == 0 ? 10 : parseInt(sumTotal) % 10; 91 var luhn = 10 - k; 92 if (lastNum == luhn) { 93 console.log("验证通过"); 94 return true; 95 } else { 96 alert("银行卡号必须符合luhn校验"); 97 return false; 98 } 99 }
再贴一个JAVA实现的LUHN算法:
1 /** 2 * 匹配luhn算法:可用于检测银行卡卡号 3 * @param cardNo 4 * @return 5 */ 6 public static boolean matchLuhn(String cardNo) { 7 int[] cardNoArr = new int[cardNo.length()]; 8 for (int i=0; i<cardNo.length(); i++) { 9 cardNoArr[i] = Integer.valueOf(String.valueOf(cardNo.charAt(i))); 10 } 11 for(int i=cardNoArr.length-2;i>=0;i-=2) { 12 cardNoArr[i] <<= 1; 13 cardNoArr[i] = cardNoArr[i]/10 + cardNoArr[i]%10; 14 } 15 int sum = 0; 16 for(int i=0;i<cardNoArr.length;i++) { 17 sum += cardNoArr[i]; 18 } 19 return sum % 10 == 0; 20 }