`
lgd_java2eye
  • 浏览: 189538 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

银联55域java解析

 
阅读更多
package com.omini.common.utils;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.omini.common.TLV;

/**
 * 本类可以把银联的55域解析成list和Map的形式
 * 
 * @author sandy
 * @version $Revision: 1.1 $ 建立日期 2012-5-7
 */
public final class SAXUnionFiled55Utils
{

	/**
	 * 银联55域
	 * 
	 * 本域将根据不同的交易种类包含不同的子域。银联处理中心仅在受理方和发卡方之间传递这些适用于IC卡交易的特有数据,而不对它们进行任何修改和处理。
	 * 为适应该子域需要不断变化的情况
	 * ,本域采用TLV(tag-length-value)的表示方式,即每个子域由tag标签(T),子域取值的长度(L)和子域取值(V)构成。
	 * tag标签的属性为bit
	 * ,由16进制表示,占1~2个字节长度。例如,"9F33"为一个占用两个字节的tag标签。而"95"为一个占用一个字节的tag标签
	 * 。若tag标签的第一个字节
	 * (注:字节排序方向为从左往右数,第一个字节即为最左边的字节。bit排序规则同理。)的后五个bit为"11111",则说明该tag占两个字节
	 * ,例如"9F33";否则占一个字节,例如"95"。 子域长度(即L本身)的属性也为bit,占1~3个字节长度。具体编码规则如下: a)
	 * 当L字段最左边字节的最左bit位(即bit8)为0,表示该L字段占一个字节,它的后续7个bit位(即bit7~bit1)表示子域取值的长度,
	 * 采用二进制数表示子域取值长度的十进制数
	 * 。例如,某个域取值占3个字节,那么其子域取值长度表示为"00000011"。所以,若子域取值的长度在1~127
	 * 字节之间,那么该L字段本身仅占一个字节。 b)
	 * 当L字段最左边字节的最左bit位(即bit8)为1,表示该L字段不止占一个字节,那么它到底占几个字节由该最左字节的后续7个bit位
	 * (即bit7~bit1)的十进制取值表示。例如,若最左字节为10000010,表示L字段除该字节外,后面还有两个字节。其后续字节
	 * 的十进制取值表示子域取值的长度。例如,若L字段为"1000 0001 1111 1111",表示该子域取值占255个字节。
	 * 所以,若子域取值的长度在128~255字节之间,那么该L字段本身需占两个字节
	 * 
	 * @return tlv list
	 */
	public static List<TLV> saxUnionField55_2List(String hexfiled55)
	{

		if (null == hexfiled55)
		{
			throw new IllegalArgumentException("55域的值不能为空!");
		}

		return builderTLV(hexfiled55);
	}

	private static List<TLV> builderTLV(String hexString)
	{
		List<TLV> tlvs = new ArrayList<TLV>();

		int position = 0;
		while (position != hexString.length())
		{
			String _hexTag = getUnionTag(hexString, position);
			position += _hexTag.length();
			
			LPositon l_position = getUnionLAndPosition(hexString, position);
			int _vl = l_position.get_vL();
			
			position = l_position.get_position();
			
			String _value = hexString.substring(position, position + _vl * 2);
			
			position = position + _value.length();
			
			tlvs.add(new TLV(_hexTag, _vl, _value));
		}
		return tlvs;
	}

	/**
	 * 银联55域
	 * 
	 * 本域将根据不同的交易种类包含不同的子域。银联处理中心仅在受理方和发卡方之间传递这些适用于IC卡交易的特有数据,而不对它们进行任何修改和处理。
	 * 为适应该子域需要不断变化的情况
	 * ,本域采用TLV(tag-length-value)的表示方式,即每个子域由tag标签(T),子域取值的长度(L)和子域取值(V)构成。
	 * tag标签的属性为bit
	 * ,由16进制表示,占1~2个字节长度。例如,"9F33"为一个占用两个字节的tag标签。而"95"为一个占用一个字节的tag标签
	 * 。若tag标签的第一个字节
	 * (注:字节排序方向为从左往右数,第一个字节即为最左边的字节。bit排序规则同理。)的后五个bit为"11111",则说明该tag占两个字节
	 * ,例如"9F33";否则占一个字节,例如"95"。 子域长度(即L本身)的属性也为bit,占1~3个字节长度。具体编码规则如下: a)
	 * 当L字段最左边字节的最左bit位(即bit8)为0,表示该L字段占一个字节,它的后续7个bit位(即bit7~bit1)表示子域取值的长度,
	 * 采用二进制数表示子域取值长度的十进制数
	 * 。例如,某个域取值占3个字节,那么其子域取值长度表示为"00000011"。所以,若子域取值的长度在1~127
	 * 字节之间,那么该L字段本身仅占一个字节。 b)
	 * 当L字段最左边字节的最左bit位(即bit8)为1,表示该L字段不止占一个字节,那么它到底占几个字节由该最左字节的后续7个bit位
	 * (即bit7~bit1)的十进制取值表示。例如,若最左字节为10000010,表示L字段除该字节外,后面还有两个字节。其后续字节
	 * 的十进制取值表示子域取值的长度。例如,若L字段为"1000 0001 1111 1111",表示该子域取值占255个字节。
	 * 所以,若子域取值的长度在128~255字节之间,那么该L字段本身需占两个字节
	 * 
	 * @return tlv map
	 */
	public static Map<String, TLV> saxUnionField55_2Map(String hexfiled55)
	{

		if (null == hexfiled55)
		{
			throw new IllegalArgumentException("55域的值不能为空!");
		}

		return builderKeyAndTLV(hexfiled55);
	}

	public static Map<String, TLV> builderKeyAndTLV(String hexString)
	{

		Map<String, TLV> tlvs = new HashMap<String, TLV>();

		int position = 0;
		while (position != hexString.length())
		{
			String _hexTag = getUnionTag(hexString, position);
			position += _hexTag.length();
			LPositon l_position = getUnionLAndPosition(hexString, position);
			int _vl = l_position.get_vL();
			position = l_position.get_position();
			String _value = hexString.substring(position, position + _vl * 2);
			position = position + _value.length();
			tlvs.put(_hexTag, new TLV(_hexTag, _vl, _value));
		}
		return tlvs;
	}

	/**
	 * 返回最后的Value的长度
	 * 
	 * @param hexString
	 * @param position
	 * @return
	 */
	private static LPositon getUnionLAndPosition(String hexString, int position)
	{

		String firstByteString = hexString.substring(position, position + 2);
		int i = Integer.parseInt(firstByteString, 16);
		String hexLength = "";

		if (((i >>> 7) & 1) == 0)
		{
			hexLength = hexString.substring(position, position + 2);
			position = position + 2;

		} else
		{
			// 当最左侧的bit位为1的时候,取得后7bit的值,
			int _L_Len = i & 127;
			position = position + 2;
			hexLength = hexString.substring(position, position + _L_Len * 2);
			// position表示第一个字节,后面的表示有多少个字节来表示后面的Value值
			position = position + _L_Len * 2;

		}
		return new LPositon(Integer.parseInt(hexLength, 16), position);

	}

	private static String getUnionTag(String hexString, int position)
	{
		String firstByte = hexString.substring(position, position + 2);
		int i = Integer.parseInt(firstByte, 16);
		if ((i & 0x1f) == 0x1f)
		{
			return hexString.substring(position, position + 4);

		} else
		{
			return hexString.substring(position, position + 2);
		}

	}

	static class LPositon
	{
		private int _vL;
		private int _position;

		public LPositon(int _vL, int position)
		{
			this._vL = _vL;
			this._position = position;
		}

		public int get_vL()
		{
			return _vL;
		}

		public void set_vL(int _vL)
		{
			this._vL = _vL;
		}

		public int get_position()
		{
			return _position;
		}

		public void set_position(int _position)
		{
			this._position = _position;
		}

	}

	public static void main(String[] args)
	{
		List<TLV> list = SAXUnionFiled55Utils
				.saxUnionField55_2List("9F260879CC8EC5A09FB9479F2701809F100807010199A0B806019F3704000000009F360201C2950500001800009A031205089C01609F02060000000000005F2A02015682027D009F1A0201569F03060000000000009F3303E0F0F09F34036003029F3501119F1E0832303033313233318405FFFFFFFFFF9F090220069F4104000000019F74064543433030319F631030313032303030308030303030303030");
		for (TLV tlv : list)
		{
			System.out.println(tlv);
		}
	}

}

 

package com.omini.common;

/**
 * @author sandy
 * @version $Revision: 1.1 $ 建立日期 2012-5-8
 */
public class TLV
{

	private String tag;

	private int length;

	private String value;

	public TLV(String tag, int length, String value)
	{
		this.length = length;
		this.tag = tag;
		this.value = value;
	}

	public String getTag()
	{
		return tag;
	}

	public void setTag(String tag)
	{
		this.tag = tag;
	}

	public int getLength()
	{
		return length;
	}

	public void setLength(int length)
	{
		this.length = length;
	}

	public String getValue()
	{
		return value;
	}

	public void setValue(String value)
	{
		this.value = value;
	}
	@Override
	public String toString()
	{
		
		return "tag=["+this.tag+"],"+"length=["+this.length+"],"+"value=["+this.value+"]";
	}

}

 

分享到:
评论
4 楼 zw1127 2013-12-09  
lgd_java2eye 写道
zw1127 写道
你的getUnionTag方法里的第三行有问题,偶数时会报错,改成if ((i & 0x0f) == 0x0f)就好了。 


改成0x0f就不对了


怎么不对了啊?求解释,我用你原来的那个方法得到的和我预期的不一样呢。
3 楼 lgd_java2eye 2013-11-22  
zw1127 写道
你的getUnionTag方法里的第三行有问题,偶数时会报错,改成if ((i & 0x0f) == 0x0f)就好了。 


改成0x0f就不对了
2 楼 zw1127 2013-10-31  
你的getUnionTag方法里的第三行有问题,偶数时会报错,改成if ((i & 0x0f) == 0x0f)就好了。 
1 楼 coolhorse168 2013-06-27  
学习了,好用,多谢

相关推荐

    iso8583实例解析银联报文(java)

    总结,"iso8583实例解析银联报文(java)"是一个基于Java的示例项目,旨在帮助开发者理解和处理银联标准下的ISO8583报文。通过这个实例,我们可以深入学习ISO8583协议的结构,掌握在Java中解析位图和字段的方法,这...

    银联iso8583简单解析实例(java)

    本文将深入探讨如何在Java环境中解析银联ISO8583报文,以及如何通过提供的代码示例`Iso8583Util.java`和`Helloworld.java`进行实践。 ISO8583协议定义了一个固定长度的报文结构,包含了多个字段,每个字段都有其...

    银联8583 java解析 最新版本 转自sourceforge

    银联 8583解析 功能很全面 支持各种数据结构

    银联8583报文解析工具

    VB做的一个比较实用的小工具,对银联终端8583报文进行解析(详细到各TLV的解析),还额外增加了相关的辅助功能,计算mac,解析清算文件等

    银联支付java 文档

    银联支付文档银联支付文档银联支付文档银联支付文档

    银联8583 java解析 1.8版

    银联 8583解析 功能很全面 支持各种数据结构 转自sourceforge

    JBIG银联电子签名解析

    3. **编程技能**:可能需要编写程序来读取、解析和解码txt文件,这可能涉及到Python、Java或其他编程语言的使用。 4. **安全处理**:在处理敏感的交易数据时,必须遵循严格的安全规范,确保数据不被泄露。 综上所述...

    银联8583报文位图解析工具

    可以解析银联8583报文各个域,根据有哪些域可以组成16进制的数,也可以进行相反的操作。

    银联电子签名解析工具

    银联电子签名解析工具是一款专为处理银联交易过程中产生的电子签名数据而设计的应用程序。在现代金融系统中,电子签名扮演着至关重要的角色,它提供了数据的安全性和交易的不可否认性。银联作为全球领先的银行卡组织...

    java版历史最全卡bin解析

    * 银联银行卡 卡bin * @author ljf */ public class UnionpayCardUtil { private static ResourceBundle bundle = ResourceBundle.getBundle("cardBin"); private static Map, String&gt; bankName = new HashMap...

    java银联多渠道ISO8583

    java实现的8583发包解包,灵活,扩展性强,8583各域能自定义编码和解码规则,包含标准的银联MAC加密算法(ansi x9.19,ansi x9.9)及一个完整的socket客户端请求服务端的demo,修改mac运算规则时,只要重写...

    chinapay 银联支付 java

    在Java环境中实现银联支付,通常涉及到一系列的技术点,包括但不限于接口调用、安全加密、交易处理和异常处理等。在这个"chinapay 银联支付 java"的项目中,我们可以深入探讨以下几个关键知识点: 1. **API接口调用...

    银联云闪付 签约 免密支付 前台版 java

    银联云闪付 下单签约-签约-支付 前台版 java程序,根据token进行免密支付

    IC卡Tag标签含义详解对照表.docx

    IC卡Tag标签含义详解对照表,对金融IC卡中各个标签的长度、格式、值、含义等以表格的形式进行了说明及解释。

    最新JAVA银联支付接口文档和示例

    JAVA最新银联支付接口文档,里面包括示例。

    银联 ISO 8583 格式数据包解析帮助工具(源码)

    通过`Java_8583_Test`文件,我们可以看到这是一个使用Java实现的测试代码,可能包含了测试用例,用于验证解析器的功能是否正确。测试通常会覆盖各种可能的数据包情况,包括正常交易、异常交易、边界条件等,以确保...

    银联代付Java代码

    总的来说,实现银联代付Java代码需要对银联接口、HTTP通信、数据解析、安全通信、异常处理等多个领域有深入理解,并且在实际操作中注意测试和合规性问题。通过熟悉提供的SDK和文档,以及实践中的不断调试优化,可以...

    银联8583报文解析和位图测试工具

    银联MIS交易调试的神器工具,不仅包含BitMap位图的解析和生成,更重要的是8583交易报文的自动解析,直接将报文复制进去,可动态解析出每个域的内容,大大提高开发测试效率

Global site tag (gtag.js) - Google Analytics