ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

go实现 银行卡Luhn校验

2022-04-20 12:31:21  阅读:113  来源: 互联网

标签:10 err 校验 字符串 cardNum go Luhn


一、银行卡号码的校验规则

银行卡号码的校验采用Luhn算法,校验过程大致如下:

1. 从右到左给卡号字符串编号,最右边第一位是1,最右边第二位是2,最右边第三位是3….

2. 从右向左遍历,对每一位字符t执行第三个步骤,并将每一位的计算结果相加得到一个数s。

3. 对每一位的计算规则:如果这一位是奇数位,则返回t本身,如果是偶数位,则先将t乘以2得到一个数n,如果n是一位数(小于10),直接返回n,否则将n的个位数和十位数相加返回。

4. 如果s能够整除10,则此号码有效,否则号码无效。

因为最终的结果会对10取余来判断是否能够整除10,所以又叫做模10算法。

二、生成符合Luhn规则的银行卡号测试数据

前面既然摸清了银行卡号的校验规则,那么就可以根据此规则生成一些能够通过Luhn校验的测试数据。

思路:

因为最右边的一位是奇数位,奇数位不需要改变值直接放啥就是啥,这个特性很重要,正好可以用来补齐到正好能够整除10。

所以显然能够推测出生成n位符合Luhn规则的算法:

1. 随机生成n-1位字符,称为字符串x。

2. 先假设字符串x有n位(实际上最右边一位缺失是n-1位,最后一位用0补上占位置),将x按照n位长度计算和s,

3. 上一步得到字符串x的校验和s,将s加上一个数字y,使得它正好可以整除10,这个y就是最右边第一位应该放的数字。

4. x+y做字符串拼接运算,得到最终的n位符合Luhn规则的字符串。

整个代码如下:

package main

import (
    "fmt"
    "math/rand"
    "strconv"
    "time"
)

func main() {
    fmt.Println(checkCarNum("6226095711989751"))
    cardNum := genCardNum("622609", 16)
    fmt.Println(cardNum)
    fmt.Println(checkCarNum(cardNum))
}

func checkCarNum(cardNum string) bool {
    sum, err := getCardNumSum(cardNum)
    if err != nil {
        return false
    }
    return sum%10 == 0
}

func getCardNumSum(cardNum string) (int64, error) {
    sum := int64(0)
    length := len(cardNum)
    index := length - 1
    for {
        t, err := strconv.ParseInt(string(cardNum[index]), 10, 64)
        if err != nil {
            return 0, err
        }

        if index%2 == 0 {
            t = t * 2
            if t >= 10 {
                t = t%10 + t/10
            }
        }

        sum += t
        if index <= 0 {
            break
        }
        index--
    }
    return sum, nil
}

func genCardNum(startWith string, totalNum int) string {
    result := startWith
    length := len(result)
    rand.New(rand.NewSource(time.Now().UnixNano()))
    for {
        result += fmt.Sprintf("%d", rand.Intn(10))
        if length == totalNum-1 {
            break
        }
        length++
    }

    sum, _ := getCardNumSum(result + "0")
    t := 10 - sum%10
    if t == 10 {
        t = 0
    }

    result += fmt.Sprintf("%d", t)
    return result
}

参考:https://www.cnblogs.com/cc11001100/p/9357177.html

标签:10,err,校验,字符串,cardNum,go,Luhn
来源: https://www.cnblogs.com/majiang/p/16169335.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有