银联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+"]";
	}

}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java解析银联55TLV(Tag,Length,Value)是指通过Java编程语言对银联报文中的55进行解析。 首先,我们需要获取整个55的值,它是一个TLV结构的数据。我们可以通过报文解析库或自己编写代码来获取55的值。 然后,我们要对55的值进行TLV解析。首先,我们要读取TLV的Tag,Tag是TLV数据的标识,它告诉我们该数据的类型或用途。我们可以从55的值中读取出Tag,一般Tag是2个字节或3个字节。 接下来,我们根据tag的长度来判断Length的长度。Length告诉我们Value的长度,它可能是1个字节,2个字节或3个字节。根据Length的长度和值,我们可以知道Value的长度。 最后,我们读取Value的值,它是Tag和Length之后的数据部分。Value可能是一个简单的数据类型,比如整数或字符串,也可能是一个组合类型的数据,比如结构体。 通过以上步骤,我们就可以解析银联55的TLV数据。我们可以将解析出的Tag、Length和Value保存到相应的数据结构中,以便后续处理和使用。 ### 回答2: TLV是Tag Length Value的缩写,是一种用于表示及解析可变长度数据的编码方式。银联55中存放了IC卡的应用数据元素(Application Data Elements,简称ADE),包括交易记录、持卡人信息、终端信息等。 Java解析银联55的TLV过程如下: 1. 获取55数据,一般以16进制字符串的形式表示。 2. 将16进制字符串转换为字节数组。 3. 遍历字节数组,根据TLV编码规则解析数据。 4. 对于每一个TLV数据,根据Tag和Length的规则解析对应的Value。 - Tag:表示数据的标签,一般为2个字节的长度。 - Length:表示数据的长度,一般为1-3个字节的长度。 - Value:表示数据的实际内容,长度由Length字段决定。 5. 根据Tag的不同,可以使用不同的算法解析Value,比如日期解析、金额解析等。 6. 解析完一个TLV数据后,可以将解析结果保存到一个数据结构中,比如Map、List等。 需要注意的是,TLV的解析过程相对复杂,需要对TLV编码规则有一定的了解。同时,解析过程中需要对字节数据进行拆分、转换和校验等操作,需要使用Java的字节处理相关的API进行操作。 总之,Java可以通过TLV编码规则解析银联55的数据,实现对IC卡应用数据元素的解析。 ### 回答3: 银联55是指银联标准规定的一个TLV(Tag-Length-Value)格式的数据,主要用于存储银行卡交易过程中的应用数据。在Java解析银联55的TLV数据,可以按照以下步骤进行: 1. 接收到55的TLV数据后,先将TLV数据转换为字节数组。 2. 对字节数组进行解析,需要按照TLV的规则来提取出每个Tag(标签)、Length(长度)和Value(值)。 3. 首先从字节数组中读取第一个字节,即Tag。根据Tag的规则,判断它的长度(一般是1-2个字节),并与下一个字节组合为一个完整的Tag。 4. 读取接下来的字节,即Length,根据Length的规则,判断它的长度(一般是1个字节或3个字节),将其转换为整数表示。 5. 根据读取到的Length,确定接下来需要读取的字节数,读取Value部分的数据。 6. 将Tag、Length和Value保存到对应的变量中,可以使用Map等数据结构来保存多个Tag对应的Value,方便后续的处理。 7. 如果还有未解析完的字节,重复步骤3-6,直到字节数组中的数据全部解析完毕。 8. 解析完成后,对获得的Tag、Length和Value进行后续的业务处理。 在解析过程中,需要注意TLV的规则以及不同数据字段的解析方式,可以参考银联的相关文档或规范来实现具体的解析逻辑。根据实际情况,可能还需要进行数据转换、格式校验、错误处理等其他的操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值