ANSI X9.8标准 PIN xor PAN获取PIN BlOCK

之前看到几篇介绍,把ANSI说成16个字节,真心扯淡,各种误人子弟,真正的ANSI算法其实是8个字节,具体格式如下:

(1) ANSI X9.8 Format(不带主账号信息)

PIN(个人识别码 Personal Identity Number)总共有8个byte长度,分为两个部分;(类似数据包的格式)

1:Byte1 记录PIN的长度

2:Byte2-Byte8 6-12位(字符)PIN(每个字符占4个BIT,不足8位右补F)

例如:明文PIN为 123456,

则PIN BLOCK为 0x06 0x12 0x34 0x56 0xFF 0xFF 0xFF 0xFF

0x06记录了PIN的长度为6,后边不足16位均以F补齐,然后转换为BCD码(BCD码为8位二进制数为一个单元,也就是一个Byte的大小也是一个十六进制数HEX的占用长度)。

2)ANSI X9.8 Format带主帐号信息)
PIN BLOCK 格式:等于 PIN 按位异或主帐号
PIN 格式:(与1中的格式类似)
Byte 1 PIN的长度
Byte 2 – Byte 3/4/5/6/7 4--12个PIN(每个PIN占4个BIT)
Byte4/5/6/7/8 – Byte 8 FILLER “F” (每个“F“占4个BIT)

PAN(主帐号 Primary Account Number)同样包含8个byte,格式如下:
Byte 1 — Byte 2 0x00 0x00
Byte 3 — Byte 8 12个主帐号字符(最后一位为校验位)
12位字符主帐号的取法:取主帐号的右12位(不包括最右边的校验位),不足12位左补“0X00”。

例子:

明文 PIN:123456,
主帐号 PAN:123456789012345678
截取下的主帐号为:678901234567 (最后一位校验位8的前12位字符为截取的主帐号)

则用于PIN加密的主帐号为:0x00 0x00 0x67 0x89 0x01 0x23 0x45 0x67
则 PIN BLOCK (PIN按位异或主帐号PAN)

即是为:  0x06 0x12 0x34 0x56 0xFF 0xFF 0xFF 0xFF
异或上:  0x00 0x00 0x67 0x89 0x01 0x23 0x45 0x67
结果为:  0x06 0x12 0x53 0xDF 0xFE 0xDC 0xBA 0x98

算法源码:(包含一个工具类Util和一个转换类ANSIFormat.java)

Util.java

package CodeApe;

public class Util {
 public Util() {

}

 
 public static void printHexString(String hint, byte[] b) {
     System.out.print(hint);
     for (int i = 0; i < b.length; i++) {
       String hex = Integer.toHexString(b[i] & 0xFF);
       if (hex.length() == 1) {
         hex = '0' + hex;
       }
       System.out.print(hex.toUpperCase() + " ");
     }
     System.out.println("");
   }

 
 
public static String Bytes2HexString(byte[] b) {
     String ret = "";
     for (int i = 0; i < b.length; i++) {
       String hex = Integer.toHexString(b[i] & 0xFF);
       if (hex.length() == 1) {
         hex = '0' + hex;
       }
       ret += hex.toUpperCase();
     }
     return ret;
   }

 
 
public static byte uniteBytes(byte src0, byte src1) {
  byte _b0 = Byte.decode("0x" + new String(new byte[] { src0 }))
    .byteValue();
  _b0 = (byte) (_b0 << 4);
  byte _b1 = Byte.decode("0x" + new String(new byte[] { src1 }))
    .byteValue();
  byte ret = (byte) (_b0 ^ _b1);
  return ret;
 }

 
 public static byte[] HexString2Bytes(String src) {
  byte[] ret = new byte[8];
  byte[] tmp = src.getBytes();
  for (int i = 0; i < 8; i++) {
  
 ret[i] = uniteBytes(tmp[i * 2], tmp[i * 2 + 1]);
  }
  return ret;
 }

}

ANSIFormat.java

package CodeApe;

import java.io.ObjectInputStream.GetField;

import javax.annotation.processing.Processor;

import CodeApe.Util;

public class ANSIFormat {

 private String pin;
 private String accno;
 public ANSIFormat(String pin , String accno){
  this.pin = pin;
  this.accno = accno;
 }
 
 public byte[] process(String pin, String accno) {
     byte arrPin[] = getHPin(pin);
     byte arrAccno[] = getHAccno(accno);
     byte arrRet[] = new byte[8];
     //PIN BLOCK 格式等于 PIN 按位异或 主帐号;
     for (int i = 0; i < 8; i++) {
       arrRet[i] = (byte) (arrPin[i] ^ arrAccno[i]);
     }
    
     Util.printHexString("PinBlock:", arrRet);
     return arrRet;
 }

 
 private byte[] getHPin(String pin) {
     byte arrPin[] = pin.getBytes();
     byte encode[] = new byte[8];
     encode[0] = (byte) 0x06;
     encode[1] = (byte) Util.uniteBytes(arrPin[0], arrPin[1]);
     encode[2] = (byte) Util.uniteBytes(arrPin[2], arrPin[3]);
     encode[3] = (byte) Util.uniteBytes(arrPin[4], arrPin[5]);
     encode[4] = (byte) 0xFF;
     encode[5] = (byte) 0xFF;
     encode[6] = (byte) 0xFF;
     encode[7] = (byte) 0xFF;
     Util.printHexString("encoded pin:", encode);
     return encode;
 }

 
 private byte[] getHAccno(String accno) {
     //取出主帐号;
     int len = accno.length();
     byte arrTemp[] = accno.substring(len < 13 ? 0 : len - 13, len - 1).getBytes();
     byte arrAccno[] = new byte[12];
     for (int i = 0; i < 12; i++) {
       arrAccno[i] = (i <= arrTemp.length ? arrTemp[i] : (byte) 0x00);
     }
     byte encode[] = new byte[8];
     encode[0] = (byte) 0x00;
     encode[1] = (byte) 0x00;
     encode[2] = (byte) Util.uniteBytes(arrAccno[0], arrAccno[1]);
     encode[3] = (byte) Util.uniteBytes(arrAccno[2], arrAccno[3]);
     encode[4] = (byte) Util.uniteBytes(arrAccno[4], arrAccno[5]);
     encode[5] = (byte) Util.uniteBytes(arrAccno[6], arrAccno[7]);
     encode[6] = (byte) Util.uniteBytes(arrAccno[8], arrAccno[9]);
     encode[7] = (byte) Util.uniteBytes(arrAccno[10], arrAccno[11]);
     Util.printHexString("encoded accno:", encode);
     return encode;
 }

 
}

test.java(测试类)

package CodeApe;

public class test {

 private static ANSIFormat pass;
 
 public static void main(String[] args) {
  // TODO Auto-generated method stub
  String pin = "123456";
  String accno = "123456789012345678";
  System.out.println("encoded pin:"+pin);
  System.out.println("encoded accno:"+accno);
  pass = new ANSIFormat(pin, accno);
  byte[] b = pass.process(pin, accno);
  
 }

}

效果图: - -!

ANSI X9.8标准 PIN xor PAN获取PIN BlOCK的更多相关文章

  1. Java实现3DES加密--及ANSI X9.8 Format标准 PIN PAN获取PIN BlOCK

    1, 采用银联ANSI X9.8标准 PIN xor PAN获取PIN BlOCK 2, 采用3Des进行加密 参考: des和3Des加密算法实现 要点:因为3DES是对称加密算法,key是24位, ...

  2. ANSI X9.19 MAC算法

    /// <summary> /// 获取MAC校验字节数据 /// </summary> /// <param name="bankData"> ...

  3. GNU C 、ANSI C、标准C、标准c++区别和联系

    转载自点击打开链接 GNU计划,又称革奴计划,是由Richard Stallman在1983年9月27日公开发起的.它的目标是创建一套完全自由的操作系统.它在编写linux的时候自己制作了一个标准成为 ...

  4. 探寻main函数的“标准”写法,以及获取main函数的参数、返回值

    main函数表示法        很多同学在初学C或者C++时,都见过各种各样的main函数表示法: main(){/*...*/} void main(){/*...*/} int main(){/ ...

  5. python 使用标准库根据进程名获取进程的pid

    有时候需要获取进程的pid,但又无法使用第三方库的时候. 方法适用linux平台. 方法1 使用subprocess 的check_output函数执行pidof命令 from subprocess ...

  6. Python 使用标准库根据进程名获取进程PID

    应用场景 在进行 Linux 运维的环境中,我们经常会遇到维护同一台服务器上的多个程序,涉及到程序的启动.关闭和重启操作. 通常这些程序之间存在着相互依存的关系需要进行依次的启动关闭操作. 下面介绍几 ...

  7. 倍福TwinCAT(贝福Beckhoff)常见问题(FAQ)-如何获取标准驱动器扭矩值获取电流值

    双击某个驱动器(以松下伺服驱动器为例),在Process Data中,注意默认显示了PDO mapping1的数据(Error code, status word等)   注意左侧,2和3分别表示了与 ...

  8. cursor pin s和cursor pin s wait on x

    1.cursor pin s是一个共享锁,一般情况下是因为发生在SQL短时间内大量执行 案例:在生产库中,突然出现大量的cursor pin s的等待,询问是否有动作后,同事说有编译存储过程(被误导了 ...

  9. LPC43xx MCU PIN Name and GPIO PIN Name Table

    //--------------------------------------------------------------------------------+ // LPC43xx Pin N ...

随机推荐

  1. inflate的使用注意事项

    public View inflate (int resource, ViewGroup root, boolean attachToRoot) 我们在使用这个方法时,要清楚原理,下面是这个方法的文档 ...

  2. <context:component-scan>

    首先看配置文件: <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http ...

  3. 经典.net面试题目(1)

    1. 简述 private. protected. public. internal 修饰符的访问权限. 答 . private :   私有成员, 在类的内部才可以访问. protected : 保 ...

  4. 关于Spring Security 3获取用户信息的问题

    标签: spring security 3标签获取用户信息 2013-01-05 10:40 5342人阅读 评论(0) 收藏 举报  分类: Spring(25) java(70) 前端(7)    ...

  5. C# 将字符串转为&#2345;这种的 html实体编码

    1.字符串转为html实体编码 private string GetHtmlEntities(string str) { string r = string.Empty; ; i < str.L ...

  6. hibernate ——联合主键

    接上一篇博客:http://www.cnblogs.com/tengpan-cn/p/5551323.html 主键类不需要写任何注解,表对象类使用@IdClass注解 在表对象类前面加@IdClas ...

  7. 运用Merge Into实现增加或更新数据

    declare @SqlStr as nvarchar(max) set @SqlStr=N'Merge Into [Categories] t USING(VALUES (9,''rice'','' ...

  8. Jenkins环境集成第一弹

    1. 起因 策划经常过来让我打包给他们测试,过于频繁影响到了自己的进度,决定弄一个打包工具让他们自己打包,在网上搜索了一下貌似有几个比较成熟的工具,Travis,Jenkins等等. 在网上也搜索到了 ...

  9. 从今日起,我会把OpenGL红宝书上的例子用完整的代码形式写在我的博客中,

    1.使用教程:OpenGL红宝书第8版 2.使用的库工具:GLEW和GLFW 3.使用的IDE:vs2012 4.说说目的:完整的看一遍OpenGL,加深印象并且熟练掌握运用OpenGL 5.欢迎有相 ...

  10. 破解MyEclipse2013注册码

    1.下载破解工具 http://down8.3987.com:801/2010/Myeclipse_zcj.3987.com.rar 2.打开 找到meclipse安装路径找到plugins文件夹打开 ...