java实现银行卡卡号 Luhm 校验算法

转自http://www.oschina.net/code/snippet_86738_3128

效验是否为银行卡,用于验证 
现行 16 位银联卡现行卡号开头 6 位是 622126~622925 之间的,7 到 15 位是银行自定义的, 
可能是发卡分行,发卡网点,发卡序号,第 16 位是校验码。 

16 位卡号校验位采用 Luhm 校验方法计算: 

1,将未带校验位的 15 位卡号从右依次编号 1 到 15,位于奇数位号上的数字乘以 2 
2,将奇位乘积的个十位全部相加,再加上所有偶数位上的数字 
3,将加法和加上校验位能被 10 整除。 
------------------------------------------ 

  
  
  
6 2 2 5 8 8 1 4 1 4 2 0 7 4 3 * 2 2 2 2 2 2 2 2 -------------------------------------------------- 12 2 4 5 16 8 2 4 2 4 4 14 4 6 将上面的数字加和:1+2+2+4+5+1+6+8+2+4+2+4+4+1+4+4+6 = 60 由于 60 加上 0 才能被 10 整除,所以校验位为 0 因此该卡号为 6225 8814 1420 7430
---------------------------------------------


如果其中一位数字换掉的话,直接导致最后校验位错误。

[java]  view plain  copy
  1. public class Test {  
  2.    
  3.     public static void main(String[] args) {  
  4.         String card = "6227007200120897790";  
  5.         System.out.println("      card: " + card);  
  6.         System.out.println("check code: " + getBankCardCheckCode(card));  
  7.         System.out.println("是否为银行卡:"+checkBankCard(card));  
  8.     }  
  9.       
  10.     /** 
  11.      * 校验银行卡卡号 
  12.      * @param cardId 
  13.      * @return 
  14.      */  
  15.     public static boolean checkBankCard(String cardId) {  
  16.              char bit = getBankCardCheckCode(cardId.substring(0, cardId.length() - 1));  
  17.              if(bit == 'N'){  
  18.                  return false;  
  19.              }  
  20.              return cardId.charAt(cardId.length() - 1) == bit;  
  21.     }  
  22.       
  23.     /** 
  24.      * 从不含校验位的银行卡卡号采用 Luhm 校验算法获得校验位 
  25.      * @param nonCheckCodeCardId 
  26.      * @return 
  27.      */  
  28.     public static char getBankCardCheckCode(String nonCheckCodeCardId){  
  29.         if(nonCheckCodeCardId == null || nonCheckCodeCardId.trim().length() == 0  
  30.                 || !nonCheckCodeCardId.matches("\\d+")) {  
  31.             //如果传的不是数据返回N  
  32.             return 'N';  
  33.         }  
  34.         char[] chs = nonCheckCodeCardId.trim().toCharArray();  
  35.         int luhmSum = 0;  
  36.         for(int i = chs.length - 1, j = 0; i >= 0; i--, j++) {  
  37.             int k = chs[i] - '0';  
  38.             if(j % 2 == 0) {  
  39.                 k *= 2;  
  40.                 k = k / 10 + k % 10;  
  41.             }  
  42.             luhmSum += k;             
  43.         }  
  44.         return (luhmSum % 10 == 0) ? '0' : (char)((10 - luhmSum % 10) + '0');  
  45.     }  
  46. }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值