8583报文组装和解析工具类(Send8583Util),代码如下:
import com.alibaba.fastjson.JSONObject;
import org.apache.commons.lang.CharEncoding;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.FileInputStream;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Field;
import java.util.*;
/**
*
* ISO8583报文工具类,包含了报文的组装和解析。
* 简单介绍下ISO8583。
* 这个东西说白了就是一种数据结构。我们定义一种规则把一堆东西放进去,再按照规则
* 把数据正确拿出来。这就是报文的实质。
* ISO8583报文的结构是:前面有16字节(128位)位图数据,后面就是数据。
* 报文最多有128个域(字段)。具体的一个报文不会有这么多,一般是几个域。
* 有哪几个就记录在位图中。而且域有定长和变长之分。
* 这些都是事先定义好的,具体可以看我写的properties定义文件.
* 位图转化成01字符串就是128个,如果某一位是1,代表这个域有值,然后按照properties定义的规则取值。
* 如果是0,则这个域没有值。
* 再说定长和变长。
* 定长域(定长比较好理解,一个字段规定是N位,那么字段值绝对不能超过N位,不足N位就在后面补空格)
* 变长域(变长域最后组装成的效果:例如变长3位,定义var3,这里的3是指长度值占3位,字段值是123456,最后结果就是006123456)
* 注意(变长的长度按照域值得字节长度计算,而不是按照域值字符串长度算!)
* 从网上不难找到ISO8583报文的介绍,这里就不多说了。
* 但是具体解析和组装的代码还真不好找,所以本人就写了一个让刚接触ISO8583报文的人更好入门。
* 解析的容器,我使用了Map,具体到工作中,还是要换成其他的容器的。
* 报文定义说明:config_8583.properties
* 例如
* FIELD031 = string,10
* FIELD032 = string,VAR2
*
* FIELD031是定长,长度是10
* FIELD032是变长,长度值占2位,也就是说长度值最大99,也就是域值最大长度99.
*/
public class Send8583Util {
private static final Logger log = LoggerFactory.getLogger(Send8583Util.class);
public static String default_encoding = CharEncoding.UTF_8;//报文编码 UTF-8 GBK
/**
* @param args
*/
public static void main(String[] args) throws Exception{
try {
String basepath=Send8583Util.class.getClassLoader().getResource("").getPath();
String path = basepath+"/config_8583.properties";
Map config8583Map = get8583Config(path);
// =====================方式二:通过数据Map组装8583报文和解析报文
System.out.println("=====================方式二:通过数据Map组装8583报文和解析报文");
//***********************组装8583报文测试--start***********************//
Map dataMap=new HashMap();//报文域
dataMap.put("flow_no", System.currentTimeMillis());
dataMap.put("trad_code", "1799");//交易码
dataMap.put("acc_no", "12345678901");//账号
dataMap.put("trad_time", "2013-11-06");//交易日期
dataMap.put("content", "aa索隆bb");//注意这个域是变长域!
dataMap.put("remark", "123456");//注意这个域是变长域!
System.out.println("初始内容报文:"+ JSONObject.toJSONString(dataMap));
String message = package8583(dataMap, config8583Map);
System.out.println("8583报文:"+message);
//***********************组装8583报文测试--end***********************//
//***********************解析8583报文测试--start***********************//
Map map=analyze8583(message, config8583Map);
System.out.println("完成解析8583报文:"+map.toString());
//***********************解析8583报文测试--end***********************//
// =====================方式三:通过域字段Map组装8583报文和解析报文
System.out.println();
System.out.println("=====================方式三:通过域字段Map组装8583报文和解析报文");
//***********************组装8583报文测试--start***********************//
TreeMap filedMap=new TreeMap();//报文域
filedMap.put("FIELD001", System.currentTimeMillis());
filedMap.put("FIELD003", "1799");//交易码
filedMap.put("FIELD008", "12345678901");//账号
filedMap.put("FIELD013", "2013-11-06");//交易日期
filedMap.put("FIELD033", "aa索隆bb");//注意这个域是变长域!
filedMap.put("FIELD036", "123456");//注意这个域是变长域!
System.out.println("初始内容报文:"+ JSONObject.toJSONString(filedMap));
byte[] send=make8583(filedMap, config8583Map);
System.out.println("最终8583报文:"+new String(send,default_encoding)+"====="+new String(send,default_encoding).length());
//***********************组装8583报文测试--end***********************//
//***********************解析8583报文测试--start***********************//
Map back=analyze8583(send, config8583Map);
System.out.println("完成解析8583报文:"+back.toString());
//***********************解析8583报文测试--end***********************//
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
}
// 8583报文初始位图:128位01字符串
public static String getInitBitMap(){
String initBitMap =
"00000000" + "00000000" + "00000000" + "00000000"
+ "00000000" + "00000000" + "00000000" + "00000000"
+ "00000000" + "00000000" + "00000000" + "00000000"
+ "00000000" + "00000000" + "00000000" + "00000000";
return initBitMap;
}
/**
* 获取8583报文定义器
* @param realPath
* @return
*/
public static Map get8583Config(String realPath) {
Properties property = new Properties();
FileInputStream fis;
try {
fis = new FileInputStream(realPath);
property.load(fis);
fis.close();
return new HashMap(property);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
* 根据实体bean和8583报文定义器-----组装8583报文
* @param bean 实体bean
* @param config8583Map 8583报文定义器
* @return
* @throws Exception
*/
public static String package8583(Object bean, Map config8583Map) throws Exception{
if(bean == null){
return null;
}
if(config8583Map == null){
return null;
}
// 从8583报文定义器中获取属性名与域字段名对应的Map
Map attributeMap = getAttributeMap(config8583Map);
if(attributeMap == null || attributeMap.size() == 0){
return null;
}
// 待组装的报文
TreeMap fieldMap = new TreeMap<>();
try {
// 遍历bean的属性
Class clazz = bean.getClass();
// 获取属性集合
Field[] fields = clazz.getDeclaredFields();
for(Field field : fields){
field.setAccessible(true); // 设置这些属性是可以访问的
String attributeName = field.getName(); // 属性名
Object value = field.get(bean); // 属性值
if(value != null && StringUtil.isNotEmpty(String.valueOf(value))){
// 如果定义器中存在改属性,则将属性名对应的域字段名与属性值放入待组装报文的Map中
if(attributeMap.containsKey(attributeName)){
fieldMap.put(attributeMap.get(attributeName), value);
}
}
}
if(fieldMap.size() > 0){
// 组装8583报文
byte[] send = make8583(fieldMap, config8583Map);
return Base64Util.encode(new String(send, CharEncoding.ISO_8859_1), CharEncoding.UTF_8);
}
}catch (Exception e){
e.printStackTrace();
}
return null;
}
/**
* 根据map和8583报文定义器-----组装8583报文
* @param dataMap 数据Map
* @param config8583Map 8583报文定义器
* @return
* @throws Exception
*/
public static String package8583(Map dataMap, Map config8583Map) throws Exception{
if(dataMap == null){
return null;
}
if(config8583Map == null){
return null;
}
// 从8583报文定义器中获取属性名与域字段名对应的Map
Map attributeMap = getAttributeMap(config8583Map);
if(attributeMap == null || attributeMap.size() == 0){
return null;
}
// 待组装的报文
TreeMap fieldMap = new TreeMap<>();
try {
// 遍历dataMap
Iterator it = dataMap.keySet().iterator();
for(;it.hasNext();){
// 属性名
String key = (String) it.next();
// 属性值
Object value = dataMap.get(key);
if(value != null && StringUtil.isNotEmpty(String.valueOf(value))){
// 如果定义器中存在改属性,则将属性名对应的域字段名与属性值放入待组装报文的Map中
if(attributeMap.containsKey(key)){
fieldMap.put(attributeMap.get(key), value);
}
}
}
if(fieldMap.size() > 0){
// 组装8583报文
byte[] send = make8583(fieldMap, config8583Map);
return Base64Util.encode(new String(send, CharEncoding.ISO_8859_1), CharEncoding.UTF_8);
}
}catch (Exception e){
e.printStackTrace();
}
return null;
}
/**
* 根据8583报文定义器-----解析8583报文
* @param message 8583报文
* @param config8583Map 8583报文定义器
* @return
* @throws Exception
*/
public static Map analyze8583(String message, Map config8583Map) throws Exception{
Map dataMap = new HashMap();
if(StringUtil.isEmpty(message)){
return null;
}
if(config8583Map == null){
return null;
}
// 从8583报文定义器中获取域字段名与属性名对应的Map
Map fieldMap = getFieldMap(config8583Map);
if(fieldMap == null || fieldMap.size() == 0){
return null;
}
try {
// 解码
message = Base64Util.decode(message, CharEncoding.UTF_8);
// 8583报文字节数组
byte[] content8583 = message.getBytes(CharEncoding.ISO_8859_1);
// 解析8583报文
Map map = analyze8583(content8583, config8583Map);
if(map != null && map.size() > 0){
Iterator it = map.keySet().iterator();
for(;it.hasNext();) {
// 域字段名
String key = (String) it.next();
// 属性值
Object value = map.get(key);
// 如果定义器中存在该域字段,则将域字段名对应的属性名与属性值放入dataMap中
if (fieldMap.containsKey(key)) {
dataMap.put(fieldMap.get(key), value);
}
}
}
return dataMap;
}catch (Exception e){
e.printStackTrace();
}
return dataMap;
}
/**
* 组装8583报文
* @param filedMap 域字段名与属性值的Map
* @param config8583Map 8583定义器
* @return
*/
public static byte[] make8583(TreeMap filedMap, Map config8583Map){
byte[] whoe8583=null;
if(filedMap==null){
return null;
}
try {
String bitMap128=getInitBitMap();//获取初始化的128位图
// 按照8583定义器格式化各个域的内容
Map all=formatValueTo8583(filedMap, bitMap128, config8583Map);
// 获取上送报文内容
whoe8583=getWhole8583Packet(all);
return whoe8583;
} catch (Exception e) {
e.printStackTrace();
}
return whoe8583;
}
/**
* 获取完整的8583报文体(128域)
* @param all
* @return
*/
public static byte[] getWhole8583Packet(Map all){
if(all==null||all.get("formatedFiledMap")==null||all.get("bitMap128")==null){
return null;
}
try {
String bitMap128=(String)all.get("bitMap128");
System.out.println("组装位图:"+bitMap128);
// 128域位图二进制字符串转16位16进制
byte[] bitmaps= get16BitByteFromStr(bitMap128);
log.info("组装位图:{}", bitMap128);
TreeMap pacBody=(TreeMap)all.get("formatedFiledMap");
StringBuffer last128=new StringBuffer();
Iterator it=pacBody.keySet().iterator();
for(;it.hasNext();){
String key=(String)it.next();
String value=(String)pacBody.get(key);
last128.append(value);
}
byte[] bitContent = last128.toString().getBytes(default_encoding);//域值
//System.out.println("数据:"+last128.toString()+"---"+last128.length());
//组装
byte[] package8583=null;
package8583= Send8583Util.arrayApend(package8583, bitmaps);
package8583= Send8583Util.arrayApend(package8583, bitContent);
return package8583;
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
/**
* 按照8583定义器格式化各个域的内容
* @param filedMap 域字段名与属性值的Map
* @param bitMap128 128位图
* @param config8583Map 8583定义器
* @return
*/
public static Map formatValueTo8583(TreeMap filedMap, String bitMap128, Map config8583Map){
Map all=new HashMap();
TreeMap formatedFiledMap=new TreeMap();//格式化结果
if(filedMap!=null){
Iterator it=filedMap.keySet().iterator();
for(;it.hasNext();){
String fieldName=(String) it.next();//例如FIELD005
String fieldValue= String.valueOf(filedMap.get(fieldName));//字段值
try{
if (fieldValue == null) {
log.error("error:报文域 {}为空值", fieldName);
return null;
}
//将域值编码转换,保证报文编码统一
fieldValue=new String(fieldValue.getBytes(default_encoding), default_encoding);
// 数据域名称FIELD开头的为128域
if (fieldName.startsWith("FIELD") && fieldValue.length() >= 0) {
String fieldNo = fieldName.substring(5, 8);//例如005
// 组二进制位图串(该域字段存在时更新128位图)
bitMap128 = change16bitMapFlag(fieldNo, bitMap128);
// 获取域定义信息
String[] fieldDef = config8583Map.get("FIELD" + fieldNo).toString().split(",");
String defType=fieldDef[0];//类型定义例string
String defLen=fieldDef[1];//长度定义,例20
boolean isFixLen=true;//是否定长判断
if(defLen.startsWith("VAR")){//变长域
isFixLen=false;
defLen=defLen.substring(3);//获取VAR后面的数字
}
// 字段值得实际长度(UTF-8下的字节长度)
int fieldLen = fieldValue.getBytes(default_encoding).length;
// 判断是否为变长域
if (!isFixLen) {
// 变长域(变长域最后组装成的效果:例如变长3位,定义var3,这里的3是指长度值占3位,字段值是123456,最后结果就是006123456)
int defLen1 = Integer.valueOf(defLen);
// 最大字节数
int maxdefLen = Integer.valueOf(strCopy("9", defLen1));
if (fieldLen > maxdefLen) {
log.error("error:字段{}的数据定义长度的长度为{}位,长度不能超过{}", fieldName, fieldLen, maxdefLen);
return null;
}else{
//将长度值组装入字段
fieldValue = getVaryLengthValue(fieldValue, defLen1) + fieldValue;
}
} else {
//定长域(定长比较好理解,一个字段规定是N位,那么字段值绝对不能超过N位,不足N位就在后面补空格)
int defLen2 = Integer.valueOf(defLen);
if (fieldLen > defLen2) {
log.error("error:字段{}的数据定义长度为{}位,长度不能超过{}", fieldName, fieldLen, defLen2);
return null;
}else{
fieldValue=getFixFieldValue(fieldValue,defLen2);//定长处理
}
}
log.info("组装后报文域 {}=={}==,域长度:{}", fieldName, fieldValue,fieldValue.getBytes(default_encoding).length);
}
// 返回结果赋值
if (filedMap.containsKey(fieldName)) {
if (formatedFiledMap.containsKey(fieldName)) {
formatedFiledMap.remove(fieldName);
}
formatedFiledMap.put(fieldName, fieldValue);
} else {
log.error("error:{}配置文件中不存在!", fieldName);
}
} catch (Exception e) {
e.printStackTrace();
}
}//end for
}
System.out.println("");
all.put("formatedFiledMap", formatedFiledMap);
all.put("bitMap128", bitMap128);
return all;
}
/**
* 解析8583报文
* @param content8583
*/
public static Map analyze8583(byte[] content8583, Map config8583Map) {
TreeMap filedMap=new TreeMap();
if(config8583Map == null){
return null;
}
try {
// 取位图
byte[] bitMap16byte = new byte[16];
System.arraycopy(content8583, 0, bitMap16byte, 0, 16);
// 16位图转2进制位图128位字符串
String bitMap128Str = get16BitMapStr(bitMap16byte);
System.out.println("解析位图:"+bitMap128Str);
log.info("解析位图:{}", bitMap128Str);
//记录当前位置,从位图后开始遍历取值
int pos = 16;
// 遍历128位图,取值。注意从FIELD001开始
for (int i = 0; i < bitMap128Str.length(); i++) {
String filedValue = "";//字段值
String filedName = "FIELD" + getNumThree((i+1)); // FIELD001
if (bitMap128Str.charAt(i) == '1') {
// 获取域定义信息
String[] fieldDef = config8583Map.get(filedName).toString().split(",");
String defType=fieldDef[0];//类型定义例string
String defLen=fieldDef[1];//长度定义,例20
boolean isFixLen=true;//是否定长判断
if(defLen.startsWith("VAR")){//变长域
isFixLen=false;
defLen=defLen.substring(3);//获取VAR2后面的2
}
// 截取该域信息
if (!isFixLen) {//变长域
int defLen1 = Integer.valueOf(defLen); //VAR后面的2
//报文中实际记录域长,例如16,023
String realLen1=new String(content8583, pos, defLen1, default_encoding);
//该字段总长度(包括长度值占的长度)
int realAllLen=defLen1+Integer.valueOf(realLen1);
byte[] filedValueByte=new byte[Integer.valueOf(realLen1)];
System.arraycopy(content8583, pos+defLen1, filedValueByte, 0, filedValueByte.length);
filedValue=new String(filedValueByte,default_encoding);
pos += realAllLen;//记录当前位置
} else {//定长域
int defLen2 = Integer.valueOf(defLen);//长度值占的位数
filedValue = new String(content8583, pos, defLen2, default_encoding).trim();
pos += defLen2;//记录当前位置
}
filedMap.put(filedName, filedValue);
}
}//end for
} catch (Exception e) {
e.printStackTrace();
}
return filedMap;
}
//********************************以下是工具方法,有些没有使用到***********************************************************//
/**
* 复制字符
* @param str
* @param count
* @return
*/
public static String strCopy(String str,int count){
StringBuffer sb = new StringBuffer();
for(int i=0;i < count;i++){
sb.append(str);
}
return sb.toString();
}
/**
* 将setContent放入set(考虑到数组越界)
* @param set
* @param setContent
* @return
*/
public static byte[] setToByte(byte[] set,byte[] setContent){
byte[] res=new byte[set.length];
if(set==null||setContent==null){
}else{
if(set.length<setContent.length){
}else{
System.arraycopy(setContent, 0, res, 0, setContent.length);
}
}
return res;
}
public static byte[] setToByte(byte[] set,String setContentStr){
byte[] res=new byte[set.length];
byte[] setContent;
try {
setContent = setContentStr.getBytes(default_encoding);
res=setToByte(res,setContent);
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return res;
}
public static String getPacketLen(int len){
String res="";
String lenStr=String.valueOf(len);
int lenC=4-lenStr.length();
res=strCopy("0",lenC)+lenStr;
return res;
}
public static String getPacketLen(String lenStr){
String res="";
if(lenStr==null){
}else{
res=getPacketLen(Integer.valueOf(lenStr));
}
return res;
}
/**
* 返回a和b的组合,实现累加功能
* @param a
* @param b
* @return
*/
public static byte[] arrayApend(byte[] a,byte[] b){
int a_len=(a==null?0:a.length);
int b_len=(b==null?0:b.length);
byte[] c=new byte[a_len+b_len];
if(a_len==0&&b_len==0){
return null;
}else if(a_len==0){
System.arraycopy(b, 0, c, 0, b.length);
}else if(b_len==0){
System.arraycopy(a, 0, c, 0, a.length);
}else{
System.arraycopy(a, 0, c, 0, a.length);
System.arraycopy(b, 0, c, a.length, b.length);
}
return c;
}
/**
* 改变128位图中的标志为1
* @param fieldNo
* @param res
* @return
*/
public static String change16bitMapFlag(String fieldNo, String res) {
int indexNo=Integer.parseInt(fieldNo);
res = res.substring(0, indexNo-1) + "1" + res.substring(indexNo);
return res;
}
/**
* 位图操作
*
* 把16位图的字节数组转化成128位01字符串
* @param bitMap16
* @return
*/
public static String get16BitMapStr(byte[] bitMap16){
String bitMap128 = "";
// 16位图转2进制位图128位字符串
for (int i = 0; i < bitMap16.length; i++) {
int bc = bitMap16[i];
bc=(bc<0)?(bc+256):bc;
String bitnaryStr=Integer.toBinaryString(bc);//二进制字符串
// 左补零,保证是8位
String rightBitnaryStr = strCopy("0",Math.abs(8-bitnaryStr.length())) + bitnaryStr;//位图二进制字符串
// 先去除多余的零,然后组装128域二进制字符串
bitMap128+=rightBitnaryStr;
}
return bitMap128;
}
/**
* 位图操作
*
* 把128位01字符串转化成16位图的字节数组
* @param str_128
* @return
*/
public static byte[] get16BitByteFromStr(String str_128){
byte[] bit16=new byte[16];
try {
if(str_128==null||str_128.length()!=128){
return null;
}
// 128域位图二进制字符串转16位16进制
byte[] tmp=str_128.getBytes(default_encoding);
int weight;//权重
byte[] strout = new byte[128];
int i, j, w = 0;
for (i = 0; i < 16; i++) {
weight = 0x0080;
for (j = 0; j < 8; j++) {
strout[i] += ((tmp[w]) - '0') * weight;
weight /= 2;
w++;
}
bit16[i] = strout[i];
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return bit16;
}
/**
* 从完整的8583报文中获取位图(16字节数组)
* @param packet
* @return
*/
public static byte[] getPacketHeaderMap(byte[] packet){
byte[] packet_header_map = new byte[16];
if(packet==null||packet.length<16){
return null;
}
for(int i=0;i<16;i++){
packet_header_map[i]=packet[i];
}
return packet_header_map;
}
/**
* 从完整的8583报文中获取16位图,转化成128位的01字符串
*
* @param content8583
* @return
*/
public static String get16BitMapFrom8583Byte(byte[] content8583){
// 取位图
byte[] bitMap16 = getPacketHeaderMap(content8583);
// 16位图转2进制位图128位字符串
String bitMap128 = get16BitMapStr(bitMap16);
return bitMap128;
}
//返回字段号码,例如005
public static String getNumThree(int i){
String len="";
String iStr=String.valueOf(i);
len=strCopy("0",3-iStr.length())+iStr;
return len;
}
/**
* 获取字符串变长值
* @param valueStr
* @param defLen
* 例如getFixLengthValue("12345678", 2)返回08
* 例如getFixLengthValue("12345678", 3)返回008
*
* 注意变长长度的计算:
* 长度的判断使用转化后的字节数组长度,因为中文在不同的编码方式下,长度是不同的,GBK是2,UTF-8是3,按字符创长度算就是1.
* 解析报文是按照字节来解析的,所以长度以字节长度为准,防止中文带来乱码。
*
* 比如一个变长域:aa索隆bb,如果按照字符串计算长度那么就是6,最后是06aa索隆bb。
* 这样在解析时按照字节解析长度就乱了,因为按照GBK字节解析,一个汉字占2,按照UTF-8解析,一个汉字占3.
* 所以在计算时必须按照字节长度为准!按照我们设置的UTF-8编码结果就是10aa索隆bb.
* 这样在解析时长度正好是10,也就不会乱了。
* @return
*/
public static String getVaryLengthValue(String valueStr,int defLen){
return getVaryLengthValue(valueStr,defLen,default_encoding);
}
public static String getVaryLengthValue(String valueStr,int defLen,String encoding){
String fixLen="";
try{
if(valueStr==null){
return strCopy("0",defLen);
}else{
byte[] valueStrByte=null;
//这里最好指定编码,不使用平台默认编码
if(encoding==null||encoding.trim().equals("")){
valueStrByte=valueStr.getBytes();
}else{
valueStrByte=valueStr.getBytes(encoding);
}
//长度的判断使用转化后的字节数组长度,因为中文在不同的编码方式下,长度是不同的,GBK是2,UTF-8是3,按字符创长度算就是1.
//解析报文是按照字节来解析的,所以长度以字节长度为准,防止中文带来乱码
// 最大字节数
int maxdefLen = Integer.valueOf(strCopy("9", defLen));
if(valueStrByte.length > maxdefLen){
return null;
}else{
int len=valueStrByte.length;//字段实际长度
String len1=String.valueOf(len);
fixLen=strCopy("0", (defLen-len1.length())) + len1;
}
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return fixLen;
}
/**
* 将字段值做定长处理,不足定长则在后面补空格
* @param valueStr
* @param defLen
* @return
*/
public static String getFixFieldValue(String valueStr,int defLen){
return getFixFieldValue(valueStr,defLen,default_encoding);
}
public static String getFixFieldValue(String valueStr,int defLen,String encoding){
String fixLen="";
try {
if(valueStr==null){
return strCopy(" ",defLen);
}else{
byte[] valueStrByte=null;
//这里最好指定编码,不使用平台默认编码
if(encoding==null||encoding.trim().equals("")){
valueStrByte=valueStr.getBytes();
}else{
valueStrByte=valueStr.getBytes(encoding);
}
//长度的判断使用转化后的字节数组长度,因为中文在不同的编码方式下,长度是不同的,GBK是2,UTF-8是3,按字符创长度算就是1.
//解析报文是按照字节来解析的,所以长度以字节长度为准,防止中文带来乱码
if(valueStrByte.length > defLen){
return null;
}else{
fixLen=valueStr+strCopy(" ",defLen-valueStrByte.length);
}
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return fixLen;
}
/**
* 获取8583报文定义器中属性名与域字段名的Map
* @param config8583Map
* @return
*/
public static Map<String, String> getAttributeMap(Map config8583Map){
Map<String, String> map = new HashMap<>();
Iterator it = config8583Map.keySet().iterator();
for(;it.hasNext();){
// 域字段名
String fieldName = (String) it.next();
// 域字段值
String fieldValue = (String) config8583Map.get(fieldName);
// 获取域定义信息
String[] fieldDef = fieldValue.split(",");
String defType = fieldDef[0]; // 类型定义,例如:string
String defLen = fieldDef[1]; // 长度定义,例如:定长 20
String attributeName = fieldDef[2]; // 报文属性名称
if(StringUtil.isNotEmpty(attributeName)){
map.put(attributeName, fieldName);
}
}
return map;
}
/**
* 获取8583报文定义器中域字段名与属性名的Map
* @param config8583Map
* @return
*/
public static Map<String, String> getFieldMap(Map config8583Map){
Map<String, String> map = new HashMap<>();
Iterator it = config8583Map.keySet().iterator();
for(;it.hasNext();){
// 域字段名
String fieldName = (String) it.next();
// 域字段值
String fieldValue = (String) config8583Map.get(fieldName);
// 获取域定义信息
String[] fieldDef = fieldValue.split(",");
String defType = fieldDef[0]; // 类型定义,例如:string
String defLen = fieldDef[1]; // 长度定义,例如:定长 20
String attributeName = fieldDef[2]; // 报文属性名称
if(StringUtil.isNotEmpty(attributeName)){
map.put(fieldName, attributeName);
}
}
return map;
}
}
8583报文定义器(config_8583.properties):
FIELD001 = string,16,flow_no
FIELD002 = string,16,card_no
FIELD003 = string,4,trad_code
FIELD008 = string,32,acc_no
FIELD013 = string,10,trad_time
FIELD033 = string,VAR2,content
FIELD036 = string,VAR3,remark
Base64Util编码解码工具:
import org.apache.commons.lang.CharEncoding;
import org.apache.commons.lang.StringUtils;
import org.apache.tomcat.util.codec.binary.Base64;
import java.io.UnsupportedEncodingException;
public class Base64Util {
/**
* 编码字符串
* @param str
* @return
*/
public static String encode(String str, String encoding){
if(StringUtils.isEmpty(str)) return "";
if(StringUtil.isEmpty(encoding)){
encoding = CharEncoding.UTF_8;
}
try {
return Base64.encodeBase64String(str.getBytes(encoding));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return "";
}
/**
* 解码字符串
* @param str
* @return
*/
public static String decode(String str, String encoding){
if(StringUtils.isEmpty(str)) return "";
if(StringUtil.isEmpty(encoding)){
encoding = CharEncoding.UTF_8;
}
try {
return new String(Base64.decodeBase64(str), encoding);
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return null;
}
}