クレジットカードの種類を番号に基づいてどのように検出しますか?


516

純粋に番号に基づいてクレジットカードの種類を検出する方法を理解しようとしています。これを見つける確実で信頼できる方法を知っている人はいますか?


3
正規表現を使用します。詳細については、このリンクを確認してください。
senfo 2008

3
詳細はすべてWikipediaにあります:en.wikipedia.org/wiki/Credit_card_numbers
Sten Vesterli

1
ウィキペディアのen.wikipedia.org/wiki/Credit_card_numbersに優れた要約表があります。カードの種類と発行者を示す最初の1〜6桁の数字です。
Alex

3
私は最初の数値グループを引き出す以外に正規表現を使用しません。通常、最初の4つの数値(米国内)からのみわかります。また、料金をクリアするために支払うことに煩わされる前に、カード番号にMod 10チェックサムを実行して、正当であることを確認してください。Luhnアルゴリズム
ダン・ブレア

3
また、これらのアルゴリズムが「いつでも」有効である場合、または「電話番号がカリフォルニアにあるかどうかを計算する」アルゴリズムのように定期的に変更する場合は、だれでもコメントできます
Simon_Weaver

回答:


772

クレジットカード/デビットカードの番号は、PANプライマリアカウント番号)と呼ばれます。PANの最初の6桁は、発行銀行に属するIINまたは発行者識別番号から取得されます(IINは以前はBIN —銀行識別番号として知られていたため、一部のドキュメントでその用語への参照が見られる場合があります)。これらの6桁は国際規格ISO / IEC 7812の対象であり、番号からカードのタイプを判別するために使用できます。

残念ながら、実際のISO / IEC 7812データベースは公開されていませんが、ウィキペディアを含め、非公式リストが商用と無料の両方で存在しています。

とにかく、数値からタイプを検出するには、次のような正規表現を使用できます。元の表現のクレジット

Visa: ^4[0-9]{6,}$ Visaカード番号は4で始まります。

MasterCard: ^5[1-5][0-9]{5,}|222[1-9][0-9]{3,}|22[3-9][0-9]{4,}|2[3-6][0-9]{5,}|27[01][0-9]{4,}|2720[0-9]{3,}$ 2016年より前は、MasterCard番号は51から55で始まりますが、これはMasterCardクレジットカードのみを検出します。このIINの範囲に該当しない、MasterCardシステムを使用して発行された他のカードがあります。2016年には、範囲(222100〜272099)の数値が追加されます。

American Express: ^3[47][0-9]{5,}$ American Expressのカード番号は34または37で始まります。

Diners Club: ^3(?:0[0-5]|[68][0-9])[0-9]{4,}$ Diners Clubカードの番号は、300から305、36、または38で始まります。5で始まり、16桁のDiners Clubカードがあります。これらはDiners ClubとMasterCardの合弁事業であり、MasterCardのように処理する必要があります。

Discover: ^6(?:011|5[0-9]{2})[0-9]{3,}$ Discoverカード番号は6011または65で始まります。

JCB: ^(?:2131|1800|35[0-9]{3})[0-9]{3,}$ JCBカードは2131、1800または35で始まります。

残念ながら、MasterCardシステムで処理されるカードの種類には、MasterCardのIIN範囲(51 ... 55で始まる番号)にないものがあります。最も重要なケースは、Maestroカードのケースです。その多くは、他の銀行のIIN範囲から発行されているため、番号空間全体に配置されています。結果として、受け入れる他のタイプではないカードはすべてMasterCardであると想定するのが最善です

重要:カード番号は長さが異なります。たとえば、Visaは過去に発行された13桁のPANを持つカードと16桁のPANを持つカードを持っています。現在、Visaのドキュメントには、12桁から19桁の数字が発行されるか、発行された可能性があることが示されています。 したがって、カード番号の長さをチェックする必要があります。ただし、カード番号が7桁以上であることを確認する必要があります(完全なIINと1つのチェックディジットの場合、これは、、Luhnアルゴリズムでたます)。

さらに1つのヒント:カード所有者のPANを処理する前に、入力から空白文字と句読点文字を削除します。どうして?通常、数字をグループで入力する方がはるかに簡単です。実際のクレジットカードの前面に数字が表示される方法と同様です。

4444 4444 4444 4444

正しく入力する方がはるかに簡単です

4444444444444444

ユーザーがここに予期しない文字を入力したため、ユーザーを懲戒しても何のメリットもありません。

これは、入力フィールドに24文字以上のスペースがあることを確認することも意味します。そうしないと、スペースを入力するユーザーはスペースを使い果たします。 32文字を表示するのに十分な幅のフィールドを作成し、最大64文字を許可することをお勧めします。これにより、拡張に十分な余裕が生まれます。

これはもう少し洞察を与える画像です:

更新(2014):チェックサム方式は、カードの信頼性を検証する有効な方法ではなくなったようですこの回答のコメントに記載されているように、。

更新(2016):マスターカードは、Ach Paymentから始まる新しいBIN範囲を実装することです。

クレジットカードの確認


7
素晴らしい例。マエストロカードの正規表現はありますか?
Manikandan 2013年

4
ダメダメダメ。カード番号の長さに依存することはできません。それらはいつでも変更できます。信頼できるカード番号の唯一の部分はIIN(これまではBINと呼ばれていました)であり、番号のプレフィックスです。さらに、Mastercardカードを推奨する方法で検出することはできません。これは、Mastercardシステムを介して処理されるカードのサブセットのみを取得します(主な問題は、さまざまなIINプレフィックスを持つMaestroカードです)。
alastair 2014

2
@alastairコメントする前に式を読みましたか?これらは特にIINを使用するように記述されているため、あなたが何を言おうとしているのか理解できません。さらに、IINを使用してカード発行者を識別できますが、検証はできません。たとえば、5412は完全なMasterCardを表すわけではありませんが、あなたの提案はそうであることを示唆しています。MasterCardが16桁以外であることの証明は見つかりませんでした。あなたの主張の情報源を自由に提供してください。ただし、Maestroカードの更新が必要であると述べているのはあなたの言うとおりです。
senfo

3
@senfoそうです、5412は完全なマスターカード番号ではありません。IINは6桁の長さであるため、完全なカード番号は7桁(最小)であり、Luhnチェックに合格する必要があります。Mastercard番号が16桁以外であることを「証明」する必要はありません。重要なのは、今日の状況に関係なく、将来的には17桁または18桁のカードを発行するか、あるいは15桁のカードを発行する可能性があるということです。
alastair 2014

3
Luhnアルゴリズムによると、一部の有効なカードに正しいチェックデジットがないとはとても信じられません。単純なタイプミスや馬鹿な詐欺の試みに対してカード番号をチェックするために、それは絶対にどこでも使用されました。代わりに、かなり賢い人がアルゴリズムを理解しておらず、間違って計算しているだけだと私は観察しました。
Rennex

74

JavaScriptで:

function detectCardType(number) {
    var re = {
        electron: /^(4026|417500|4405|4508|4844|4913|4917)\d+$/,
        maestro: /^(5018|5020|5038|5612|5893|6304|6759|6761|6762|6763|0604|6390)\d+$/,
        dankort: /^(5019)\d+$/,
        interpayment: /^(636)\d+$/,
        unionpay: /^(62|88)\d+$/,
        visa: /^4[0-9]{12}(?:[0-9]{3})?$/,
        mastercard: /^5[1-5][0-9]{14}$/,
        amex: /^3[47][0-9]{13}$/,
        diners: /^3(?:0[0-5]|[68][0-9])[0-9]{11}$/,
        discover: /^6(?:011|5[0-9]{2})[0-9]{12}$/,
        jcb: /^(?:2131|1800|35\d{3})\d{11}$/
    }

    for(var key in re) {
        if(re[key].test(number)) {
            return key
        }
    }
}

単体テスト:

describe('CreditCard', function() {
    describe('#detectCardType', function() {

        var cards = {
            '8800000000000000': 'UNIONPAY',

            '4026000000000000': 'ELECTRON',
            '4175000000000000': 'ELECTRON',
            '4405000000000000': 'ELECTRON',
            '4508000000000000': 'ELECTRON',
            '4844000000000000': 'ELECTRON',
            '4913000000000000': 'ELECTRON',
            '4917000000000000': 'ELECTRON',

            '5019000000000000': 'DANKORT',

            '5018000000000000': 'MAESTRO',
            '5020000000000000': 'MAESTRO',
            '5038000000000000': 'MAESTRO',
            '5612000000000000': 'MAESTRO',
            '5893000000000000': 'MAESTRO',
            '6304000000000000': 'MAESTRO',
            '6759000000000000': 'MAESTRO',
            '6761000000000000': 'MAESTRO',
            '6762000000000000': 'MAESTRO',
            '6763000000000000': 'MAESTRO',
            '0604000000000000': 'MAESTRO',
            '6390000000000000': 'MAESTRO',

            '3528000000000000': 'JCB',
            '3589000000000000': 'JCB',
            '3529000000000000': 'JCB',

            '6360000000000000': 'INTERPAYMENT',

            '4916338506082832': 'VISA',
            '4556015886206505': 'VISA',
            '4539048040151731': 'VISA',
            '4024007198964305': 'VISA',
            '4716175187624512': 'VISA',

            '5280934283171080': 'MASTERCARD',
            '5456060454627409': 'MASTERCARD',
            '5331113404316994': 'MASTERCARD',
            '5259474113320034': 'MASTERCARD',
            '5442179619690834': 'MASTERCARD',

            '6011894492395579': 'DISCOVER',
            '6011388644154687': 'DISCOVER',
            '6011880085013612': 'DISCOVER',
            '6011652795433988': 'DISCOVER',
            '6011375973328347': 'DISCOVER',

            '345936346788903': 'AMEX',
            '377669501013152': 'AMEX',
            '373083634595479': 'AMEX',
            '370710819865268': 'AMEX',
            '371095063560404': 'AMEX'
        };

        Object.keys(cards).forEach(function(number) {
            it('should detect card ' + number + ' as ' + cards[number], function() {
                Basket.detectCardType(number).should.equal(cards[number]);
            });
        });
    });
});

1
@ jolly.exe-すべてのテストでフィドルがundefinedを返します。動作しません:(
ShadeTreeDeveloper

@ShadeTreeDeveloperは任意の値を入力するだけです。372176090165471(テキストフィールドのAMAX)
Code Spy

@ jolly.exeわかりました...(キーアップイベントをオフにして)入力時にフォーマットされるものを期待していました。整数を入力してもフィドルは機能します。
ShadeTreeDeveloper 2015

結局、必要な入力のフォーマットと検証を行うために、このコードを書きました。quercusv.github.io/smartForm
ShadeTreeDeveloper 2015

あなたはv-payとbancontactカード番号を検出する方法を知っていますか?ありがとう
Oleksandr IY 2018

38

更新日:2016年6月15日(現在、最終的な解決策として)

投票数が上位に投票された場合でも、私は投票を放棄することに注意してください。ただし、これらの正規表現が実際に機能することを明確にするために、何千もの実際のBINコードでテストしました。最も重要なのは、開始文字列(^)を使用することです。そうしないと、現実の世界で誤った結果が返されます。

JCB ^(?:2131|1800|35)[0-9]{0,}$開始:2131、1800、35(3528-3589)

アメリカンエクスプレス ^3[47][0-9]{0,}$34、37

Diners Club ^3(?:0[0-59]{1}|[689])[0-9]{0,}$開始:300-305、309、36、38-39

ビザ ^4[0-9]{0,}$開始:4

MasterCard ^(5[1-5]|222[1-9]|22[3-9]|2[3-6]|27[01]|2720)[0-9]{0,}$開始:2221-2720、51-55

マエストロ ^(5[06789]|6)[0-9]{0,}$マエストロは常に範囲で成長しています: 60〜69他のものではなく/で始まりますが、とにかく5から始まるものはマスターカードとしてエンコードする必要があります。Maestroカードはコードの最後で検出する必要があります。他のカードは60〜69の範囲にあるためです。コードを見てください。

発見する ^(6011|65|64[4-9]|62212[6-9]|6221[3-9]|622[2-8]|6229[01]|62292[0-5])[0-9]{0,}$コードに非常に困難を発見、で始まる:6011、622126から622925まで、644から649まで、65

JavaScriptの私は、この関数を使用します。これは、それをonkeyupイベントに割り当てて、できるだけ早く結果を出すときに役立ちます。

function cc_brand_id(cur_val) {
    // the regular expressions check for possible matches as you type, hence the OR operators based on the number of chars
    // regexp string length {0} provided for soonest detection of beginning of the card numbers this way it could be used for BIN CODE detection also

    //JCB
    jcb_regex = new RegExp('^(?:2131|1800|35)[0-9]{0,}$'); //2131, 1800, 35 (3528-3589)
    // American Express
    amex_regex = new RegExp('^3[47][0-9]{0,}$'); //34, 37
    // Diners Club
    diners_regex = new RegExp('^3(?:0[0-59]{1}|[689])[0-9]{0,}$'); //300-305, 309, 36, 38-39
    // Visa
    visa_regex = new RegExp('^4[0-9]{0,}$'); //4
    // MasterCard
    mastercard_regex = new RegExp('^(5[1-5]|222[1-9]|22[3-9]|2[3-6]|27[01]|2720)[0-9]{0,}$'); //2221-2720, 51-55
    maestro_regex = new RegExp('^(5[06789]|6)[0-9]{0,}$'); //always growing in the range: 60-69, started with / not something else, but starting 5 must be encoded as mastercard anyway
    //Discover
    discover_regex = new RegExp('^(6011|65|64[4-9]|62212[6-9]|6221[3-9]|622[2-8]|6229[01]|62292[0-5])[0-9]{0,}$');
    ////6011, 622126-622925, 644-649, 65


    // get rid of anything but numbers
    cur_val = cur_val.replace(/\D/g, '');

    // checks per each, as their could be multiple hits
    //fix: ordering matter in detection, otherwise can give false results in rare cases
    var sel_brand = "unknown";
    if (cur_val.match(jcb_regex)) {
        sel_brand = "jcb";
    } else if (cur_val.match(amex_regex)) {
        sel_brand = "amex";
    } else if (cur_val.match(diners_regex)) {
        sel_brand = "diners_club";
    } else if (cur_val.match(visa_regex)) {
        sel_brand = "visa";
    } else if (cur_val.match(mastercard_regex)) {
        sel_brand = "mastercard";
    } else if (cur_val.match(discover_regex)) {
        sel_brand = "discover";
    } else if (cur_val.match(maestro_regex)) {
        if (cur_val[0] == '5') { //started 5 must be mastercard
            sel_brand = "mastercard";
        } else {
            sel_brand = "maestro"; //maestro is all 60-69 which is not something else, thats why this condition in the end
        }
    }

    return sel_brand;
}

ここでそれを試すことができます:

http://jsfiddle.net/upN3L/69/

PHPでこの関数を使用すると、サブVISA / MCカードも検出されます。

/**
  * Obtain a brand constant from a PAN
  *
  * @param string $pan               Credit card number
  * @param bool   $include_sub_types Include detection of sub visa brands
  * @return string
  */
public static function getCardBrand($pan, $include_sub_types = false)
{
    //maximum length is not fixed now, there are growing number of CCs has more numbers in length, limiting can give false negatives atm

    //these regexps accept not whole cc numbers too
    //visa
    $visa_regex = "/^4[0-9]{0,}$/";
    $vpreca_regex = "/^428485[0-9]{0,}$/";
    $postepay_regex = "/^(402360|402361|403035|417631|529948){0,}$/";
    $cartasi_regex = "/^(432917|432930|453998)[0-9]{0,}$/";
    $entropay_regex = "/^(406742|410162|431380|459061|533844|522093)[0-9]{0,}$/";
    $o2money_regex = "/^(422793|475743)[0-9]{0,}$/";

    // MasterCard
    $mastercard_regex = "/^(5[1-5]|222[1-9]|22[3-9]|2[3-6]|27[01]|2720)[0-9]{0,}$/";
    $maestro_regex = "/^(5[06789]|6)[0-9]{0,}$/";
    $kukuruza_regex = "/^525477[0-9]{0,}$/";
    $yunacard_regex = "/^541275[0-9]{0,}$/";

    // American Express
    $amex_regex = "/^3[47][0-9]{0,}$/";

    // Diners Club
    $diners_regex = "/^3(?:0[0-59]{1}|[689])[0-9]{0,}$/";

    //Discover
    $discover_regex = "/^(6011|65|64[4-9]|62212[6-9]|6221[3-9]|622[2-8]|6229[01]|62292[0-5])[0-9]{0,}$/";

    //JCB
    $jcb_regex = "/^(?:2131|1800|35)[0-9]{0,}$/";

    //ordering matter in detection, otherwise can give false results in rare cases
    if (preg_match($jcb_regex, $pan)) {
        return "jcb";
    }

    if (preg_match($amex_regex, $pan)) {
        return "amex";
    }

    if (preg_match($diners_regex, $pan)) {
        return "diners_club";
    }

    //sub visa/mastercard cards
    if ($include_sub_types) {
        if (preg_match($vpreca_regex, $pan)) {
            return "v-preca";
        }
        if (preg_match($postepay_regex, $pan)) {
            return "postepay";
        }
        if (preg_match($cartasi_regex, $pan)) {
            return "cartasi";
        }
        if (preg_match($entropay_regex, $pan)) {
            return "entropay";
        }
        if (preg_match($o2money_regex, $pan)) {
            return "o2money";
        }
        if (preg_match($kukuruza_regex, $pan)) {
            return "kukuruza";
        }
        if (preg_match($yunacard_regex, $pan)) {
            return "yunacard";
        }
    }

    if (preg_match($visa_regex, $pan)) {
        return "visa";
    }

    if (preg_match($mastercard_regex, $pan)) {
        return "mastercard";
    }

    if (preg_match($discover_regex, $pan)) {
        return "discover";
    }

    if (preg_match($maestro_regex, $pan)) {
        if ($pan[0] == '5') { //started 5 must be mastercard
            return "mastercard";
        }
        return "maestro"; //maestro is all 60-69 which is not something else, thats why this condition in the end

    }

    return "unknown"; //unknown for this system
}

1
また、これはCC番号の検出のみであり、検証ではないことに注意してください。それは分離されており、Luhnチェックである必要があります...
Janos Szabo

Visa Electronはどこにありますか。また、Maestroの小切手がMasterCardを返す場合があるのはなぜですか?MasterCardはそれ自体をチェックしませんか?
BadHorsie 16

このJCBテスト番号をどのタイプ(3088514174175777)としても認識せず、このテストJCB番号をdiners_club(3096278649822922)として識別します。テストカード番号のリストをしていると仮定すると、有効なとにかく(freeformatter.com/credit-card-number-generator-validator.html
ドリュー

308または309からJCBカードになる可能性があるというドキュメントはありません
Janos Szabo

ccタイプ検出コードを提供するための+1。これは、通常uxに対して実行したいことです。MCの新しい範囲の正規表現には、小さな週が必要です:/ ^(5 [1-5] | 222 [1-9] | 22 [3-9] [0-9] | 2 [3-6] [0-9] {2} | 27 [01] [0-9] | 2720)[0-9] {0、} $ /
kinakuta 16

21
public string GetCreditCardType(string CreditCardNumber)
{
    Regex regVisa = new Regex("^4[0-9]{12}(?:[0-9]{3})?$");
    Regex regMaster = new Regex("^5[1-5][0-9]{14}$");
    Regex regExpress = new Regex("^3[47][0-9]{13}$");
    Regex regDiners = new Regex("^3(?:0[0-5]|[68][0-9])[0-9]{11}$");
    Regex regDiscover = new Regex("^6(?:011|5[0-9]{2})[0-9]{12}$");
    Regex regJCB = new Regex("^(?:2131|1800|35\\d{3})\\d{11}$");


    if (regVisa.IsMatch(CreditCardNumber))
        return "VISA";
    else if (regMaster.IsMatch(CreditCardNumber))
        return "MASTER";
    else  if (regExpress.IsMatch(CreditCardNumber))
        return "AEXPRESS";
    else if (regDiners.IsMatch(CreditCardNumber))
        return "DINERS";
    else if (regDiscover.IsMatch(CreditCardNumber))
        return "DISCOVERS";
    else if (regJCB.IsMatch(CreditCardNumber))
        return "JCB";
    else
        return "invalid";
}

Regexを使用してクレジットカードの種類を確認する関数は次のとおりです。c#


19

これをチェックしてください:

http://www.breakingpar.com/bkp/home.nsf/0/87256B280015193F87256CC70060A01B

function isValidCreditCard(type, ccnum) {
    /* Visa: length 16, prefix 4, dashes optional.
    Mastercard: length 16, prefix 51-55, dashes optional.
    Discover: length 16, prefix 6011, dashes optional.
    American Express: length 15, prefix 34 or 37.
    Diners: length 14, prefix 30, 36, or 38. */

    var re = new Regex({
        "visa": "/^4\d{3}-?\d{4}-?\d{4}-?\d",
        "mc": "/^5[1-5]\d{2}-?\d{4}-?\d{4}-?\d{4}$/",
        "disc": "/^6011-?\d{4}-?\d{4}-?\d{4}$/",
        "amex": "/^3[47]\d{13}$/",
        "diners": "/^3[068]\d{12}$/"
    }[type.toLowerCase()])

    if (!re.test(ccnum)) return false;
    // Remove all dashes for the checksum checks to eliminate negative numbers
    ccnum = ccnum.split("-").join("");
    // Checksum ("Mod 10")
    // Add even digits in even length strings or odd digits in odd length strings.
    var checksum = 0;
    for (var i = (2 - (ccnum.length % 2)); i <= ccnum.length; i += 2) {
        checksum += parseInt(ccnum.charAt(i - 1));
    }
    // Analyze odd digits in even length strings or even digits in odd length strings.
    for (var i = (ccnum.length % 2) + 1; i < ccnum.length; i += 2) {
        var digit = parseInt(ccnum.charAt(i - 1)) * 2;
        if (digit < 10) { checksum += digit; } else { checksum += (digit - 9); }
    }
    if ((checksum % 10) == 0) return true;
    else return false;
}

15

最近、私はそのような機能が必要でした、私はZend Framework Credit Card Validatorをルビーに移植していました。ruby gem:https : //github.com/Fivell/credit_card_validations zend framework: https //github.com/zendframework/zf2/blob/master/library/Zend/Validator/CreditCard.php

どちらもINN範囲を使用してタイプを検出します。ここであなたは読むことができます INNについて

これによると、クレジットカードを代わりに検出できます(正規表現なしで、プレフィックスと可能な長さに関するいくつかのルールを宣言します)

ほとんどの中古カードには次のルールがあります

########  most used brands #########

    visa: [
        {length: [13, 16], prefixes: ['4']}
    ],
    mastercard: [
        {length: [16], prefixes: ['51', '52', '53', '54', '55']}
    ],

    amex: [
        {length: [15], prefixes: ['34', '37']}
    ],
    ######## other brands ########
    diners: [
        {length: [14], prefixes: ['300', '301', '302', '303', '304', '305', '36', '38']},
    ],

    #There are Diners Club (North America) cards that begin with 5. These are a joint venture between Diners Club and MasterCard, and are processed like a MasterCard
    # will be removed in next major version

    diners_us: [
        {length: [16], prefixes: ['54', '55']}
    ],

    discover: [
        {length: [16], prefixes: ['6011', '644', '645', '646', '647', '648',
                                  '649', '65']}
    ],

    jcb: [
        {length: [16], prefixes: ['3528', '3529', '353', '354', '355', '356', '357', '358', '1800', '2131']}
    ],


    laser: [
        {length: [16, 17, 18, 19], prefixes: ['6304', '6706', '6771']}
    ],

    solo: [
        {length: [16, 18, 19], prefixes: ['6334', '6767']}
    ],

    switch: [
        {length: [16, 18, 19], prefixes: ['633110', '633312', '633304', '633303', '633301', '633300']}

    ],

    maestro: [
        {length: [12, 13, 14, 15, 16, 17, 18, 19], prefixes: ['5010', '5011', '5012', '5013', '5014', '5015', '5016', '5017', '5018',
                                                              '502', '503', '504', '505', '506', '507', '508',
                                                              '6012', '6013', '6014', '6015', '6016', '6017', '6018', '6019',
                                                              '602', '603', '604', '605', '6060',
                                                              '677', '675', '674', '673', '672', '671', '670',
                                                              '6760', '6761', '6762', '6763', '6764', '6765', '6766', '6768', '6769']}
    ],

    # Luhn validation are skipped for union pay cards because they have unknown generation algoritm
    unionpay: [
        {length: [16, 17, 18, 19], prefixes: ['622', '624', '625', '626', '628'], skip_luhn: true}
    ],

    dankrot: [
        {length: [16], prefixes: ['5019']}
    ],

    rupay: [
        {length: [16], prefixes: ['6061', '6062', '6063', '6064', '6065', '6066', '6067', '6068', '6069', '607', '608'], skip_luhn: true}
    ]

}

次に、プレフィックスを検索して長さを比較することにより、クレジットカードのブランドを検出できます。また、luhnアルゴリズムについても忘れないでください(http://en.wikipedia.org/wiki/Luhnで説明されています))。

更新

ルールの更新されたリストは、https://raw.githubusercontent.com/Fivell/credit_card_validations/master/lib/data/brands.yamlにあります。


2
非常にわかりやすい。VISAカードの長さは13桁です。
Herman Kan

@HermanKan、何のVISAのウェブサイトは、それは16の長さでなければなりません言わない、私はそれが13あることではなく、今日でした長い時間前に考える
Fivell

1
レガシーサポートだと思います
Fivell

1
@HermanKan、もう1つあります。VISAにはVPayカードがあり、Wikipediaに準拠してVisaのVPayブランドは13から19桁のPANの長さを指定できるため、16桁を超えるカード番号が表示されています。
Fivell、2014


13

以下は、codeproject上のすべての種類のCC関連完全なC#またはVBコードです

  • IsValidNumber
  • GetCardTypeFromNumber
  • GetCardTestNumber
  • PasssLuhnTest

この記事は数年前からあり、否定的なコメントはありません。


1
@barett-修正しました。「aspnet」カテゴリから「検​​証」カテゴリに移動し、リンクが変更されたようです
Simon_Weaver

2
リンクが壊れています。多分これは同じユーティリティですか?codeproject.com/Articles/20271/...
ジョシュ・ノエ

そのcodeprojectコードは2007年のものです。警告、古い可能性があります。
アロン

8

コンパクトなJavaScriptバージョン

    var getCardType = function (number) {
        var cards = {
            visa: /^4[0-9]{12}(?:[0-9]{3})?$/,
            mastercard: /^5[1-5][0-9]{14}$/,
            amex: /^3[47][0-9]{13}$/,
            diners: /^3(?:0[0-5]|[68][0-9])[0-9]{11}$/,
            discover: /^6(?:011|5[0-9]{2})[0-9]{12}$/,
            jcb: /^(?:2131|1800|35\d{3})\d{11}$/
        };
        for (var card in cards) {
            if (cards[card].test(number)) {
                return card;
            }
        }
    };

8

PHPにおけるAnatoliyの回答:

 public static function detectCardType($num)
 {
    $re = array(
        "visa"       => "/^4[0-9]{12}(?:[0-9]{3})?$/",
        "mastercard" => "/^5[1-5][0-9]{14}$/",
        "amex"       => "/^3[47][0-9]{13}$/",
        "discover"   => "/^6(?:011|5[0-9]{2})[0-9]{12}$/",
    );

    if (preg_match($re['visa'],$num))
    {
        return 'visa';
    }
    else if (preg_match($re['mastercard'],$num))
    {
        return 'mastercard';
    }
    else if (preg_match($re['amex'],$num))
    {
        return 'amex';
    }
    else if (preg_match($re['discover'],$num))
    {
        return 'discover';
    }
    else
    {
        return false;
    }
 }

7

これは、PHPクラス関数がCCtypeをCCnumberで返すPHPクラス関数です。
このコードは、カードを検証しないか、Luhnアルゴリズムを実行しません。このページの表に基づいて、クレジットカードの種類を検索するだけです。基本的に、CCnumberタイプとCCcardプレフィックスを使用してCCcardタイプを決定します。

<?php
class CreditcardType
{
    public static $creditcardTypes = [
        [
            'Name' => 'American Express',
            'cardLength' => [15],
            'cardPrefix' => ['34', '37'],
        ], [
            'Name' => 'Maestro',
            'cardLength' => [12, 13, 14, 15, 16, 17, 18, 19],
            'cardPrefix' => ['5018', '5020', '5038', '6304', '6759', '6761', '6763'],
        ], [
            'Name' => 'Mastercard',
            'cardLength' => [16],
            'cardPrefix' => ['51', '52', '53', '54', '55'],
        ], [
            'Name' => 'Visa',
            'cardLength' => [13, 16],
            'cardPrefix' => ['4'],
        ], [
            'Name' => 'JCB',
            'cardLength' => [16],
            'cardPrefix' => ['3528', '3529', '353', '354', '355', '356', '357', '358'],
        ], [
            'Name' => 'Discover',
            'cardLength' => [16],
            'cardPrefix' => ['6011', '622126', '622127', '622128', '622129', '62213','62214', '62215', '62216', '62217', '62218', '62219','6222', '6223', '6224', '6225', '6226', '6227', '6228','62290', '62291', '622920', '622921', '622922', '622923','622924', '622925', '644', '645', '646', '647', '648','649', '65'],
        ], [
            'Name' => 'Solo',
            'cardLength' => [16, 18, 19],
            'cardPrefix' => ['6334', '6767'],
        ], [
            'Name' => 'Unionpay',
            'cardLength' => [16, 17, 18, 19],
            'cardPrefix' => ['622126', '622127', '622128', '622129', '62213', '62214','62215', '62216', '62217', '62218', '62219', '6222', '6223','6224', '6225', '6226', '6227', '6228', '62290', '62291','622920', '622921', '622922', '622923', '622924', '622925'],
        ], [
            'Name' => 'Diners Club',
            'cardLength' => [14],
            'cardPrefix' => ['300', '301', '302', '303', '304', '305', '36'],
        ], [
            'Name' => 'Diners Club US',
            'cardLength' => [16],
            'cardPrefix' => ['54', '55'],
        ], [
            'Name' => 'Diners Club Carte Blanche',
            'cardLength' => [14],
            'cardPrefix' => ['300', '305'],
        ], [
            'Name' => 'Laser',
            'cardLength' => [16, 17, 18, 19],
            'cardPrefix' => ['6304', '6706', '6771', '6709'],
        ],
    ];

    public static function getType($CCNumber)
    {
        $CCNumber = trim($CCNumber);
        $type = 'Unknown';
        foreach (CreditcardType::$creditcardTypes as $card) {
            if (! in_array(strlen($CCNumber), $card['cardLength'])) {
                continue;
            }
            $prefixes = '/^(' . implode('|', $card['cardPrefix']) . ')/';
            if (preg_match($prefixes, $CCNumber) == 1) {
                $type = $card['Name'];
                break;
            }
        }
        return $type;
    }
}

6

支払い処理の一環としてクレジットカードの種類を検出しないでください。有効なトランザクションを拒否するリスクがあります。

支払い処理業者に情報を提供する必要がある場合(たとえば、PayPalクレジットカードオブジェクトはカードタイプに名前を付ける必要があります)、使用可能な最小の情報から推測します(例:

$credit_card['pan'] = preg_replace('/[^0-9]/', '', $credit_card['pan']);
$inn = (int) mb_substr($credit_card['pan'], 0, 2);

// @see http://en.wikipedia.org/wiki/List_of_Bank_Identification_Numbers#Overview
if ($inn >= 40 && $inn <= 49) {
    $type = 'visa';
} else if ($inn >= 51 && $inn <= 55) {
    $type = 'mastercard';
} else if ($inn >= 60 && $inn <= 65) {
    $type = 'discover';
} else if ($inn >= 34 && $inn <= 37) {
    $type = 'amex';
} else {
    throw new \UnexpectedValueException('Unsupported card type.');
}

この実装(最初の2桁のみを使用)は、すべての主要な(そしてPayPalの場合はサポートされているすべての)カードスキームを識別するのに十分です。実際、例外をすべてスキップして、最も一般的なカードタイプにデフォルト設定することをお勧めします。リクエストへの応答として検証エラーがあるかどうかを支払いゲートウェイ/プロセッサに通知させます。

現実には、支払いゲートウェイは提供する価値を気にしません


1
これは単に真実ではありません。カードの種類を渡す必要がある3つの異なるプロバイダーを知っています。渡さない場合、トランザクションは失敗します。
Ed DeGagne 2014

3
@EdDeGagne-「どの値を気にしない」は「渡されても気にしない」と同じではありません。
Quentin Skousen、2014

どちらを指定しましたか?私は、CCタイプで渡すことをYOUに要求する使用中のプロバイダーがあることだけを述べましたが、それ以上のものはありません。
Ed DeGagne 2014

この複雑な問題を単純化することはできませんが、通常、支払いプロバイダーはカードタイプを提案する必要がなく、検出する独自の方法があります
Janos Szabo

6

クレジットカードの最初の数字は、ベンダーを概算するために使用できます。

  • ビザ:49、44、47
  • ビザの電子:42、45、48、49
  • マスターカード:51
  • アメックス:34
  • ディナー:30、36、38
  • JCB:35

これらの範囲は大幅に更新されており、カードベンダー企業は投稿に記載されているよりも多くの範囲を追加しています。
NJInamdar

6

カード範囲認識(CRR)で、一連の正規表現または他のハードコードされた範囲を使用するアルゴリズムの欠点は、BIN / IINが時間の経過とともに変化することです。カードのブランド提携は現在進行中の問題です。異なるカード取得者/商人は、たとえば地理位置情報に応じて、同じカードを異なる方法で処理する必要がある場合があります。

さらに、UnionPayカードなどの普及が進んだ過去数年間、既存のモデルは、新しい範囲に取って代わり、より広い範囲にインターリーブする場合があります。
一部の範囲は特定の国での使用に制限されているため、システムがカバーする必要がある地理を知ることが役立つ場合があります。たとえば、範囲62には米国の一部のAAAサブ範囲が含まれていますが、販売拠点が米国外にある場合は、62をすべてUnionPayとして扱うことができる場合があります。
また、販売者の場所に基づいてカードの扱いを変えるように求められる場合もあります。たとえば、特定の英国のカードを国内ではデビットカードとして、国際的にはクレジットカードとして扱います。

1つの主要なアクワイアリングバンクによって維持されている非常に便利な一連のルールがあります。例:https : //www.barclaycard.co.uk/business/files/BIN-Rules-EIRE.pdfおよびhttps://www.barclaycard.co.uk/business/files/BIN-Rules-UK.pdf。(2017年6月現在、有効なリンクは、更新された参照へのリンクを提供したユーザーのおかげです。)ただし、これらのCRRルールは、そのエンティティによって取得されたマーチャントに適用されるため、カード発行ユニバースを表す場合があることに注意してください。たとえば、CUP / UPIとして識別される範囲は含まれません。

これらのコメントは、磁気ストライプ(MagStripe)またはPKE(Pan Key Entry)シナリオに適用されます。ICC / EMVの世界でも状況は異なります。

更新:このページ(およびリンクされているWikiPediaページも)の他の回答には、JCBが常に16桁あります。ただし、私の会社には、複数の買収銀行や地域にわたってPOSデバイスとソフトウェアを認定する専門のエンジニアチームがあります。このチームがJCBから持っているカードの最新の認定パックには、19の長いPANのパスケースがありました。


こんにちは@CaiqueOliveira、更新されたリンクを参照してください。更新されたBINルールのリファレンスへのリンクを提供してくれたmac9416に感謝します。
MikeRoger 2017年

1
更新されたBIN-Rulesリファレンスについては@ mac9416に感謝します。
MikeRoger 2017年

5

Usman Yの回答のSwift 2.1バージョン。印刷ステートメントを使用して検証するので、文字列値で呼び出します

print(self.validateCardType(self.creditCardField.text!))

func validateCardType(testCard: String) -> String {

    let regVisa = "^4[0-9]{12}(?:[0-9]{3})?$"
    let regMaster = "^5[1-5][0-9]{14}$"
    let regExpress = "^3[47][0-9]{13}$"
    let regDiners = "^3(?:0[0-5]|[68][0-9])[0-9]{11}$"
    let regDiscover = "^6(?:011|5[0-9]{2})[0-9]{12}$"
    let regJCB = "^(?:2131|1800|35\\d{3})\\d{11}$"


    let regVisaTest = NSPredicate(format: "SELF MATCHES %@", regVisa)
    let regMasterTest = NSPredicate(format: "SELF MATCHES %@", regMaster)
    let regExpressTest = NSPredicate(format: "SELF MATCHES %@", regExpress)
    let regDinersTest = NSPredicate(format: "SELF MATCHES %@", regDiners)
    let regDiscoverTest = NSPredicate(format: "SELF MATCHES %@", regDiscover)
    let regJCBTest = NSPredicate(format: "SELF MATCHES %@", regJCB)


    if regVisaTest.evaluateWithObject(testCard){
        return "Visa"
    }
    else if regMasterTest.evaluateWithObject(testCard){
        return "MasterCard"
    }

    else if regExpressTest.evaluateWithObject(testCard){
        return "American Express"
    }

    else if regDinersTest.evaluateWithObject(testCard){
        return "Diners Club"
    }

    else if regDiscoverTest.evaluateWithObject(testCard){
        return "Discover"
    }

    else if regJCBTest.evaluateWithObject(testCard){
        return "JCB"
    }

    return ""

}

4

Stripeは、カードスキーム検出用にこの素晴らしいJavaScriptライブラリを提供しています。コードスニペットをいくつか追加して、その使用方法を示します。

最初にそれをあなたのウェブページに含めてください

<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jquery.payment/1.2.3/jquery.payment.js " ></script>

次に、カードスキームを検出するために関数cardTypeを使用します。

$(document).ready(function() {              
            var type = $.payment.cardType("4242 4242 4242 4242"); //test card number
            console.log(type);                                   
}); 

その他の例とデモの参照リンクを以下に示します。

  1. jquery.payment.jsのストライプブログ
  2. Githubリポジトリ

4

迅速に、クレジットカードの種類を検出する列挙型を作成できます。

enum CreditCardType: Int { // Enum which encapsulates different card types and method to find the type of card.

case Visa
case Master
case Amex
case Discover

func validationRegex() -> String {
    var regex = ""
    switch self {
    case .Visa:
        regex = "^4[0-9]{6,}$"

    case .Master:
        regex = "^5[1-5][0-9]{5,}$"

    case .Amex:
        regex = "^3[47][0-9]{13}$"

    case .Discover:
        regex = "^6(?:011|5[0-9]{2})[0-9]{12}$"
    }

    return regex
}

func validate(cardNumber: String) -> Bool {
    let predicate = NSPredicate(format: "SELF MATCHES %@", validationRegex())
    return predicate.evaluateWithObject(cardNumber)
}

// Method returns the credit card type for given card number
static func cardTypeForCreditCardNumber(cardNumber: String) -> CreditCardType?  {
    var creditCardType: CreditCardType?

    var index = 0
    while let cardType = CreditCardType(rawValue: index) {
        if cardType.validate(cardNumber) {
            creditCardType = cardType
            break
        } else {
            index++
        }
    }
    return creditCardType
  }
}

CreditCardType列挙値を返すメソッドCreditCardType.cardTypeForCreditCardNumber( "#card number")を呼び出します。


3

jQueryを使用した私のソリューション:

function detectCreditCardType() {
    var type = new Array;
    type[1] = '^4[0-9]{12}(?:[0-9]{3})?$';      // visa
    type[2] = '^5[1-5][0-9]{14}$';              // mastercard
    type[3] = '^6(?:011|5[0-9]{2})[0-9]{12}$';  // discover
    type[4] = '^3[47][0-9]{13}$';               // amex

    var ccnum = $('.creditcard').val().replace(/[^\d.]/g, '');
    var returntype = 0;

    $.each(type, function(idx, re) {
        var regex = new RegExp(re);
        if(regex.test(ccnum) && idx>0) {
            returntype = idx;
        }
    });

    return returntype;
}

0が返された場合、クレジットカードの種類は検出されません。

「クレジットカード」クラスをクレジットカード入力フィールドに追加する必要があります。


1
既存の回答のバリエーション。
Gajus 2014

1
はい、上記の回答のコードを使用して、改善し、ここに投稿しました。反対票をありがとう...
ZurabWeb 2014

3
(a)これを既存のコードの改善として提案する、(b)適切な貢献を書く、または(c)正規表現の作成に使用したソースを参照する必要があります。
Gajus 2014

1
ガジュス、私はその瞬間に私がコミュニティを助けたと信じています。誰かのために何かをした方がいいと言わないでください。私は役に立ったかもしれないけれども私はやったことをしました。
ZurabWeb 2014

3

私はクレジットカードのフォーマットと電話番号のフォーマットをかなり調べました。たくさんの良いヒントが見つかりましたが、私の希望にぴったり合うものは何もなかったので、このコードを作成しました。次のように使用します。

var sf = smartForm.formatCC(myInputString);
var cardType = sf.cardType;

2
// abobjects.com, parvez ahmad ab bulk mailer
use below script

function isValidCreditCard2(type, ccnum) {
       if (type == "Visa") {
          // Visa: length 16, prefix 4, dashes optional.
          var re = /^4\d{3}?\d{4}?\d{4}?\d{4}$/;
       } else if (type == "MasterCard") {
          // Mastercard: length 16, prefix 51-55, dashes optional.
          var re = /^5[1-5]\d{2}?\d{4}?\d{4}?\d{4}$/;
       } else if (type == "Discover") {
          // Discover: length 16, prefix 6011, dashes optional.
          var re = /^6011?\d{4}?\d{4}?\d{4}$/;
       } else if (type == "AmEx") {
          // American Express: length 15, prefix 34 or 37.
          var re = /^3[4,7]\d{13}$/;
       } else if (type == "Diners") {
          // Diners: length 14, prefix 30, 36, or 38.
          var re = /^3[0,6,8]\d{12}$/;
       }
       if (!re.test(ccnum)) return false;
       return true;
       /*
       // Remove all dashes for the checksum checks to eliminate negative numbers
       ccnum = ccnum.split("-").join("");
       // Checksum ("Mod 10")
       // Add even digits in even length strings or odd digits in odd length strings.
       var checksum = 0;
       for (var i=(2-(ccnum.length % 2)); i<=ccnum.length; i+=2) {
          checksum += parseInt(ccnum.charAt(i-1));
       }
       // Analyze odd digits in even length strings or even digits in odd length strings.
       for (var i=(ccnum.length % 2) + 1; i<ccnum.length; i+=2) {
          var digit = parseInt(ccnum.charAt(i-1)) * 2;
          if (digit < 10) { checksum += digit; } else { checksum += (digit-9); }
       }
       if ((checksum % 10) == 0) return true; else return false;
       */

    }
jQuery.validator.addMethod("isValidCreditCard", function(postalcode, element) { 
    return isValidCreditCard2($("#cardType").val(), $("#cardNum").val()); 

}, "<br>credit card is invalid");


     Type</td>
                                          <td class="text">&nbsp; <form:select path="cardType" cssclass="fields" style="border: 1px solid #D5D5D5;padding: 0px 0px 0px 0px;width: 130px;height: 22px;">
                                              <option value="SELECT">SELECT</option>
                                              <option value="MasterCard">Mastercard</option>
                                              <option value="Visa">Visa</option>
                                               <option value="AmEx">American Express</option>
                                              <option value="Discover">Discover</option>
                                            </form:select> <font color="#FF0000">*</font> 

$("#signupForm").validate({

    rules:{
       companyName:{required: true},
       address1:{required: true},
       city:{required: true},
       state:{required: true},
       zip:{required: true},
       country:{required: true},
       chkAgree:{required: true},
       confPassword:{required: true},
       lastName:{required: true},
       firstName:{required: true},
       ccAddress1:{required: true},
       ccZip:{         
           postalcode : true
       },
       phone:{required: true},
       email:{
           required: true,
           email: true
           },
       userName:{
           required: true,
           minlength: 6
           },
       password:{
           required: true,
           minlength: 6
           },          
       cardNum:{           
            isValidCreditCard : true
       },

問題は、特定の実装ではなく、クレジットカードをチェックするアルゴリズムについてです。このコードは何をしますか?
EmilVikström、2012年

2

ほんの少しのスプーンフィーディング:

$("#CreditCardNumber").focusout(function () {


        var regVisa = /^4[0-9]{12}(?:[0-9]{3})?$/;
        var regMasterCard = /^5[1-5][0-9]{14}$/;
        var regAmex = /^3[47][0-9]{13}$/;
        var regDiscover = /^6(?:011|5[0-9]{2})[0-9]{12}$/;

        if (regVisa.test($(this).val())) {
            $("#CCImage").html("<img height='40px' src='@Url.Content("~/images/visa.png")'>");          

        }

        else if (regMasterCard.test($(this).val())) {
        $("#CCImage").html("<img height='40px' src='@Url.Content("~/images/mastercard.png")'>");

        }

        else if (regAmex.test($(this).val())) {

           $("#CCImage").html("<img height='40px' src='@Url.Content("~/images/amex.png")'>");

        }
         else if (regDiscover.test($(this).val())) {

           $("#CCImage").html("<img height='40px' src='@Url.Content("~/images/discover.png")'>");

        }
        else {
        $("#CCImage").html("NA");

        }

    });

2

True関数名ごとにカードが検出された場合に返される、Pythonで記述されたいくつかのブール関数の例を次に示します。

def is_american_express(cc_number):
    """Checks if the card is an american express. If us billing address country code, & is_amex, use vpos
    https://en.wikipedia.org/wiki/Bank_card_number#cite_note-GenCardFeatures-3
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^3[47][0-9]{13}$', cc_number))


def is_visa(cc_number):
    """Checks if the card is a visa, begins with 4 and 12 or 15 additional digits.
    :param cc_number: unicode card number
    """

    # Standard Visa is 13 or 16, debit can be 19
    if bool(re.match(r'^4', cc_number)) and len(cc_number) in [13, 16, 19]:
        return True

    return False


def is_mastercard(cc_number):
    """Checks if the card is a mastercard. Begins with 51-55 or 2221-2720 and 16 in length.
    :param cc_number: unicode card number
    """
    if len(cc_number) == 16 and cc_number.isdigit():  # Check digit, before cast to int
        return bool(re.match(r'^5[1-5]', cc_number)) or int(cc_number[:4]) in range(2221, 2721)
    return False


def is_discover(cc_number):
    """Checks if the card is discover, re would be too hard to maintain. Not a supported card.
    :param cc_number: unicode card number
    """
    if len(cc_number) == 16:
        try:
            # return bool(cc_number[:4] == '6011' or cc_number[:2] == '65' or cc_number[:6] in range(622126, 622926))
            return bool(cc_number[:4] == '6011' or cc_number[:2] == '65' or 622126 <= int(cc_number[:6]) <= 622925)
        except ValueError:
            return False
    return False


def is_jcb(cc_number):
    """Checks if the card is a jcb. Not a supported card.
    :param cc_number: unicode card number
    """
    # return bool(re.match(r'^(?:2131|1800|35\d{3})\d{11}$', cc_number))  # wikipedia
    return bool(re.match(r'^35(2[89]|[3-8][0-9])[0-9]{12}$', cc_number))  # PawelDecowski


def is_diners_club(cc_number):
    """Checks if the card is a diners club. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^3(?:0[0-6]|[68][0-9])[0-9]{11}$', cc_number))  # 0-5 = carte blance, 6 = international


def is_laser(cc_number):
    """Checks if the card is laser. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^(6304|670[69]|6771)', cc_number))


def is_maestro(cc_number):
    """Checks if the card is maestro. Not a supported card.
    :param cc_number: unicode card number
    """
    possible_lengths = [12, 13, 14, 15, 16, 17, 18, 19]
    return bool(re.match(r'^(50|5[6-9]|6[0-9])', cc_number)) and len(cc_number) in possible_lengths


# Child cards

def is_visa_electron(cc_number):
    """Child of visa. Checks if the card is a visa electron. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^(4026|417500|4508|4844|491(3|7))', cc_number)) and len(cc_number) == 16


def is_total_rewards_visa(cc_number):
    """Child of visa. Checks if the card is a Total Rewards Visa. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^41277777[0-9]{8}$', cc_number))


def is_diners_club_carte_blanche(cc_number):
    """Child card of diners. Checks if the card is a diners club carte blance. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^30[0-5][0-9]{11}$', cc_number))  # github PawelDecowski, jquery-creditcardvalidator


def is_diners_club_carte_international(cc_number):
    """Child card of diners. Checks if the card is a diners club international. Not a supported card.
    :param cc_number: unicode card number
    """
    return bool(re.match(r'^36[0-9]{12}$', cc_number))  # jquery-creditcardvalidator

1

カード番号の最初の6桁(最初のMII桁を含む)は、発行者識別番号(IIN)と呼ばれます。これらは、カード所有者にカードを発行したカード発行機関を識別します。残りの番号はカード発行会社によって割り当てられます。カード番号の長さは、その桁数です。多くのカード発行会社は、カードにIIN全体と口座番号を印刷しています。

上記の事実に基づいて、カードのブランドを識別するためにJAVAコードのスニペットを保持したいと思います。

カードの種類のサンプル

public static final String AMERICAN_EXPRESS = "American Express";
public static final String DISCOVER = "Discover";
public static final String JCB = "JCB";
public static final String DINERS_CLUB = "Diners Club";
public static final String VISA = "Visa";
public static final String MASTERCARD = "MasterCard";
public static final String UNKNOWN = "Unknown";

カードのプレフィックス

// Based on http://en.wikipedia.org/wiki/Bank_card_number#Issuer_identification_number_.28IIN.29
public static final String[] PREFIXES_AMERICAN_EXPRESS = {"34", "37"};
public static final String[] PREFIXES_DISCOVER = {"60", "62", "64", "65"};
public static final String[] PREFIXES_JCB = {"35"};
public static final String[] PREFIXES_DINERS_CLUB = {"300", "301", "302", "303", "304", "305", "309", "36", "38", "39"};
public static final String[] PREFIXES_VISA = {"4"};
public static final String[] PREFIXES_MASTERCARD = {
        "2221", "2222", "2223", "2224", "2225", "2226", "2227", "2228", "2229",
        "223", "224", "225", "226", "227", "228", "229",
        "23", "24", "25", "26",
        "270", "271", "2720",
        "50", "51", "52", "53", "54", "55"
    };

入力番号に指定されたプレフィックスのいずれかがあるかどうかを確認してください。

public String getBrand(String number) {

String evaluatedType;
if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_AMERICAN_EXPRESS)) {
    evaluatedType = AMERICAN_EXPRESS;
} else if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_DISCOVER)) {
    evaluatedType = DISCOVER;
} else if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_JCB)) {
    evaluatedType = JCB;
} else if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_DINERS_CLUB)) {
    evaluatedType = DINERS_CLUB;
} else if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_VISA)) {
    evaluatedType = VISA;
} else if (StripeTextUtils.hasAnyPrefix(number, PREFIXES_MASTERCARD)) {
    evaluatedType = MASTERCARD;
} else {
    evaluatedType = UNKNOWN;
}
    return evaluatedType;
}

最後に、Utilityメソッド

/**
  * Check to see if the input number has any of the given prefixes.
  *
  * @param number the number to test
  * @param prefixes the prefixes to test against
  * @return {@code true} if number begins with any of the input prefixes
*/

public static boolean hasAnyPrefix(String number, String... prefixes) {
  if (number == null) {
       return false;
  }
   for (String prefix : prefixes) {
       if (number.startsWith(prefix)) {
       return true;
    }
  }
     return false;
}

参照


1

これをコトリンに試してください。Regexを追加し、whenステートメントに追加します。

private fun getCardType(number: String): String {

        val visa = Regex("^4[0-9]{12}(?:[0-9]{3})?$")
        val mastercard = Regex("^5[1-5][0-9]{14}$")
        val amx = Regex("^3[47][0-9]{13}$")

        return when {
            visa.matches(number) -> "Visa"
            mastercard.matches(number) -> "Mastercard"
            amx.matches(number) -> "American Express"
            else -> "Unknown"
        }
    }

0

それぞれのカードベンダーに一致する正規表現ルール:

  • (4\d{12}(?:\d{3})?) VISAの場合。
  • (5[1-5]\d{14}) マスターカード用。
  • (3[47]\d{13}) AMEX用。
  • ((?:5020|5038|6304|6579|6761)\d{12}(?:\d\d)?) マエストロのために。
  • (3(?:0[0-5]|[68][0-9])[0-9]{11}) ダイナースクラブのために。
  • (6(?:011|5[0-9]{2})[0-9]{12}) 発見のために。
  • (35[2-8][89]\d\d\d{10}) JCBの場合。

JCBの正規表現は間違っていると思います。たとえば、3528から3589までの最初の4桁はすべて受け入れられますが、3570は受け入れられません。
Gabe

0

https://github.com/bendrucker/creditcards-types/を使用して、クレジットカードの種類を番号から検出します。私が遭遇した1つの問題は、テスト番号6011 1111 1111 1117を発見することです

https://www.cybersource.com/developers/other_resources/quick_references/test_cc_numbers/から 6011で始まるため、それは検出番号であることがわかります。しかし、クレジットカードタイプから取得した結果は「Maestro」です。著者に問題を公開しました。彼はすぐに私に返信して、このPDFドキュメントを提供しました。https://www.discovernetwork.com/downloads/IPP_VAR_Compliance.pdfこのドキュメント から、6011 1111 1111 1117がDiscoverクレジットカードの範囲に入らないことがはっきりとわかります。


私は同じ問題を抱えていますが、これを修正しましたか?
lucasvm1980 2017

@ lucasvm1980 discovernetwork.com pdfファイルの方が信頼性が高いと思います。そして、6011 1111 1111 1117という数字は単なるテスト用の数字であり、実際のクレジットカードにはこの数字はありません。だから私はこれを心配する必要はないと思います。
yuxiaomin 2017

Discoverカードにバグがあるようです。有効な番号を試しましたが、そのエラーも発生しました。
lucasvm1980 2017

@ lucasvm1980番号を提供してgithubに問題を送信できますか?
yuxiaomin 2017

0

これを試してください。

func checkCardValidation(number : String) -> Bool
{
    let reversedInts = number.characters.reversed().map { Int(String($0)) }
        return reversedInts.enumerated().reduce(0, {(sum, val) in
            let odd = val.offset % 2 == 1
            return sum + (odd ? (val.element! == 9 ? 9 : (val.element! * 2) % 9) : val.element!)
        }) % 10 == 0
}

使用する。

if (self.checkCardValidation(number: "yourNumber") == true) {
     print("Card Number valid")
}else{
     print("Card Number not valid")
}

0
follow Luhn’s algorithm

 private  boolean validateCreditCardNumber(String str) {

        int[] ints = new int[str.length()];
        for (int i = 0; i < str.length(); i++) {
            ints[i] = Integer.parseInt(str.substring(i, i + 1));
        }
        for (int i = ints.length - 2; i >= 0; i = i - 2) {
            int j = ints[i];
            j = j * 2;
            if (j > 9) {
                j = j % 10 + 1;
            }
            ints[i] = j;
        }
        int sum = 0;
        for (int i = 0; i < ints.length; i++) {
            sum += ints[i];
        }
        if (sum % 10 == 0) {
           return true;
        } else {
            return false;
        }


    }

then call this method

Edittext mCreditCardNumberEt;

 mCreditCardNumberEt.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {

             int cardcount=   s.toString().length();
                 if(cardcount>=16) {
                    boolean cardnumbervalid=   validateCreditCardNumber(s.toString());
                    if(cardnumbervalid) {
                        cardvalidtesting.setText("Valid Card");
                        cardvalidtesting.setTextColor(ContextCompat.getColor(context,R.color.green));
                    }
                    else {
                        cardvalidtesting.setText("Invalid Card");
                        cardvalidtesting.setTextColor(ContextCompat.getColor(context,R.color.red));
                    }
                }
               else if(cardcount>0 &&cardcount<16) {
                     cardvalidtesting.setText("Invalid Card");
                     cardvalidtesting.setTextColor(ContextCompat.getColor(context,R.color.red));
                }

                else {
                    cardvalidtesting.setText("");

                }


                }

            @Override
            public void afterTextChanged(Editable s) {

            }
        });
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.