第一次对接这玩意,也是RLG!!! 对接硬件不都用C来搞吗?没法奈何我只会java
不废话,先直接上部分对接需求文档>>>
先解读一下以上的东西,看备注可知,以0x 开头,基本确定是全是16进制的东西
先理一下,
类型为Byte, 一个字节等于2位16进制数
类型为Uint ...反正跟java中的"int"八九不离十, int占4个字节等于8位16进制数
类型为Ushort ...反正跟java中的"short八九不离十, short占2个字节等于4位16进制数
还有个要注意的点,数字类型高低位排序。低位在“前”按照甲方爸爸给的提示,这个“前”是从右往左排 ,意思是越小的,越排在后面
以长度lenth举个栗子吧:
//当长度 lenth=18时
int lenth=18;
//转16进制
String lenthHex = Integer.toHexString(lenth);
//这时的lenthHex应该等于12,但是文档要求长度为4x2位16进制,于是
//往前补0,这是的lenthHex等于00000012
//再低位在前排序,注意,此处全部都是这对于字节,于是2位一体倒序
//输出lenthHex=12000000
下面给出 设备登录的需求文档来做一次实际操作
//设备登陆测试
final static String info =
"01" +//开始标记
"41000000" +//长度 LEN 原值为65 (低位排序)
"00000000" +//
"00000000" +//
"01" +//版本
"4B03" +//命令 原值为843 (低位排序)
"1B00ED8B91D23143AFE845EAA4F7EFCF" +//唯一标识 (随便填的,文档只要求保证唯一性)
//content 原值为 厂家识别码:15EFD2B2FD6B476697B729F33D08630A+设备的唯一标识码:5078c650bac04a0fa190bb7480573f26 +xor运算可参考 http://www.ip33.com/bcc.html
"31354546443242324644364234373636393742373239463333443038363330413530373863363530626163303461306661313930626237343830353733663236" +"05" +//xor运算
"00" +//状态
"01";//结束标记
来看一下测试工具的返回结果
返回Flag区域=00表示处理成功
下面给出java代码中如何实现发送和接收数据代码,基于springboot,为了性能全局只创建一次连接
数据发送者
/**
* 在项目启动完成后启动 优先级1,发送数据直接调用sendMsg方法即可
*/
@Component
@Order(value = 1)
public class BaseServer implements ApplicationRunner {
private final static Logger logger = LoggerFactory.getLogger(AttendanceServer.class);
private static final ThreadLocal<Socket> threadConnect = new ThreadLocal<Socket>();
//对方域名
private static final String HOST = "";
//对方监听端口
private final static int PORT = 8080;
//服务输出流
public static OutputStream outStr = null;
//服务输入流
public static InputStream inStr = null;
//Socket连接
public static Socket client;
public void run(ApplicationArguments args) throws Exception {
try {
//初始化连接
this.connect();
}catch (Exception e){
logger.error("初始化连接服务出现异常!"+e.getMessage());
e.printStackTrace();
}
}
/**
* 连接初始化方法
* @throws Exception
*/
public static void connect() throws Exception {
client = threadConnect.get();
//如果client为空,则建立一次新的连接
if (client == null) {
client = new Socket(HOST, PORT);
threadConnect.set(client);
logger.info("服务初始化成功!");
}
outStr = client.getOutputStream();
inStr = client.getInputStream();
}
/**
* 关闭连接方法
* @throws Exception
*/
public static void disconnect() {
try {
outStr.close();
inStr.close();
client.close();
outStr=null;
inStr=null;
client=null;
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 发送数据方法(添加同步锁,防止输出流被写入垃圾数据)
* @param info
*/
public synchronized static void sendMsg(String info) {
byte[] m = HexUtil.hexStringToBytes(info);
try {
logger.info("准备发送报文:"+info);
Thread.sleep(1500);
outStr.write(m);
outStr.flush();
} catch (Exception e) {
e.printStackTrace();
//如果此处捕获到异常,直接关闭当前连接,然后重新初始化
logger.info("发送报文出现异常,即将重新建立连接!");
disconnect();
try {
//初始化连接
connect();
}catch (Exception ex){
logger.error("初始化连接服务出现异常!"+ex.getMessage());
ex.printStackTrace();
}
}
}
}
接收者服务
/**
* 数据接收服务 ,继承自连接服务
* 在项目启动完成后启动 优先级2
*/
@Component
@Order(value = 2)
public class Receiver extends BaseServer implements ApplicationRunner {
private final Logger logger = LoggerFactory.getLogger(Receiver.class);
@Override
public void run(ApplicationArguments args) throws Exception {
logger.info("数据接收服务初始化成功!");
try {
while (true) {
byte[] b = new byte[1024];
if (inStr == null) {
//可能是连接还在初始化中,需要等待
continue;
}
Thread.sleep(500);
if (inStr.available() > 0) {
int r = inStr.read(b);
if (r > -1) {
String result = HexUtil.BinaryToHexString(b);
}
}
}
} catch (IOException e) {
logger.error("数据接收服务出现异常" + e.getMessage() + "原因是:");
e.printStackTrace();
}
}
}
进制转换工具类
public class HexUtil {
public static byte[] hexStringToBytes(String hexString) {
if (hexString == null || hexString.equals("")) {
return null;
}
// toUpperCase将字符串中的所有字符转换为大写
hexString = hexString.toUpperCase();
int length = hexString.length() / 2;
// toCharArray将此字符串转换为一个新的字符数组。
char[] hexChars = hexString.toCharArray();
byte[] d = new byte[length];
for (int i = 0; i < length; i++) {
int pos = i * 2;
d[i] = (byte) (charToByte(hexChars[pos]) << 4 | charToByte(hexChars[pos + 1]));
}
return d;
}
//charToByte返回在指定字符的第一个发生的字符串中的索引,即返回匹配字符
private static byte charToByte(char c) {
return (byte) "0123456789ABCDEF".indexOf(c);
}
public static String BinaryToHexString(byte[] bytes) {
String hexStr = "0123456789ABCDEF";
String result = "";
String hex = "";
for (byte b : bytes) {
hex = String.valueOf(hexStr.charAt((b & 0xF0) >> 4));
hex += String.valueOf(hexStr.charAt(b & 0x0F));
result += hex ;
}
return result;
}
//两位一字符,倒序排序
public static String reverseString(String str) {
List<String> strlist=new ArrayList();
char[] chr = str.toCharArray();
for (int i = 0 ; i < chr.length; i=i+2) {
String s=chr[i]+""+chr[i+1];
strlist.add(s);
}
Collections.reverse(strlist);
String result="";
for(String v:strlist){
result+=v;
}
return result;
}
/**
* 16进制转换成为string类型字符串
* @param s
* @return
*/
public static String hexStringToString(String s) {
if (s == null || s.equals("")) {
return null;
}
s = s.replace(" ", "");
byte[] baKeyword = new byte[s.length() / 2];
for (int i = 0; i < baKeyword.length; i++) {
try {
baKeyword[i] = (byte) (0xff & Integer.parseInt(s.substring(i * 2, i * 2 + 2), 16));
} catch (Exception e) {
e.printStackTrace();
}
}
try {
s = new String(baKeyword, "UTF-8");
new String();
} catch (Exception e1) {
e1.printStackTrace();
}
return s;
}
/**
* 字符串转化成为16进制字符串
* @param s
* @return
*/
public static String strTo16(String s) {
String str = "";
for (int i = 0; i < s.length(); i++) {
int ch = (int) s.charAt(i);
String s4 = Integer.toHexString(ch);
str = str + s4;
}
return str;
}
//将16进制字符串自动补全到8位 并且倒序排序
public static String full8(String lenth) {
int a = lenth.getBytes().length;
int b = 8 - a;
for (int i = 0; i < b; i++) {
lenth = "0" + lenth;
}
return reverseString(lenth);
}
/**
* xor运算
*
* @param data
* @return
*/
public static String getBCC(byte[] data) {
String ret = "";
byte BCC[] = new byte[1];
for (int i = 0; i < data.length; i++) {
BCC[0] = (byte) (BCC[0] ^ data[i]);
}
String hex = Integer.toHexString(BCC[0] & 0xFF);
if (hex.length() == 1) {
hex = '0' + hex;
}
ret += hex.toUpperCase();
return ret;
}
}
稍微修改一下就能用了
来源:oschina
链接:https://my.oschina.net/u/4295888/blog/3339986