有时候在网上办理一些业务时有些须要填写银行卡号码,当胡乱填写时会当即报错,可是并无发现向后端发送请求,那么这个效果是怎么实现的呢。php
对于银行卡号有一个校验算法,叫作Luhn算法。html
银行卡号码的校验采用Luhn算法,校验过程大体以下:前端
1. 从右到左给卡号字符串编号,最右边第一位是1,最右边第二位是2,最右边第三位是3….python
2. 从右向左遍历,对每一位字符t执行第三个步骤,并将每一位的计算结果相加获得一个数s。算法
3. 对每一位的计算规则:若是这一位是奇数位,则返回t自己,若是是偶数位,则先将t乘以2获得一个数n,若是n是一位数(小于10),直接返回n,不然将n的个位数和十位数相加返回。后端
4. 若是s可以整除10,则此号码有效,不然号码无效。dom
由于最终的结果会对10取余来判断是否可以整除10,因此又叫作模10算法。工具
校验算法比较简单,一个python的实现:测试
#! /usr/bin/python3 # -*- coding: utf-8 -*- def luhn(card_num): s = 0 card_num_length = len(card_num) for _ in range(1, card_num_length + 1): t = int(card_num[card_num_length - _]) if _ % 2 == 0: t *= 2 s += t if t < 10 else t % 10 + t // 10 else: s += t return s % 10 == 0 if __name__ == '__main__': print(luhn('6226095711989751'))
前面既然摸清了银行卡号的校验规则,那么就能够根据此规则生成一些可以经过Luhn校验的测试数据。.net
思路:
由于最右边的一位是奇数位,奇数位不须要改变值直接放啥就是啥,这个特性很重要,正好能够用来补齐到正好可以整除10。
因此显然可以推测出生成n位符合Luhn规则的算法:
1. 随机生成n-1位字符,称为字符串x。
2. 先假设字符串x有n位(实际上最右边一位缺失是n-1位),将x按照n位长度计算和s,由于最右边第一位是缺失的,忽略跳过,因此计算时最右边一位从2开始。
3. 上一步获得字符串x的校验和s,将s加上一个数字y,使得它正好能够整除10,这个y就是最右边第一位应该放的数字。
4. x+y作字符串拼接运算,获得最终的n位符合Luhn规则的字符串。
实现代码:
#! /usr/bin/python3 # -*- coding: utf-8 -*- import random def gen_card_num(start_with, total_num): result = start_with # 随机生成前N-1位 while len(result) < total_num - 1: result += str(random.randint(0, 9)) # 计算前N-1位的校验和 s = 0 card_num_length = len(result) for _ in range(2, card_num_length + 2): t = int(result[card_num_length - _ + 1]) if _ % 2 == 0: t *= 2 s += t if t < 10 else t % 10 + t // 10 else: s += t # 最后一位当作是校验位,用来补齐到可以整除10 t = 10 - s % 10 result += str(0 if t == 10 else t) return result def luhn(card_num): s = 0 card_num_length = len(card_num) for _ in range(1, card_num_length + 1): t = int(card_num[card_num_length - _]) if _ % 2 == 0: t *= 2 s += t if t < 10 else t % 10 + t // 10 else: s += t return s % 10 == 0 if __name__ == '__main__': for _ in range(1000): random_card_num = gen_card_num('622609', 16) valid_result = luhn(random_card_num) print('%s %s' % (random_card_num, valid_result))
1. 在开发须要填写银行卡号的表单时,最好可以在前端加上一层Luhn校验,以将大部分的非法输入在前端就拦截过滤掉。
2. 在须要一些银行卡号测试数据时,可使用上面的代码生成一些合法的银行卡号做为测试数据。
3. 明白了这些以后之后转帐再输卡号不用那么担忧了,由于若是不当心输错了一位的话可以校验出来的,固然理论上是这样的,但对于我这样的穷人十块钱以上的高额交易就得确认好几回…
相关资料:
.