Как читать данные сетевого потока в Java (протокол пакетов)
Я работаю над декодированием протокола, используемого определенным устройством GPS для связи. Прямо сейчас я анализирую первый пакет данных, который он отправляет. Я умею читать, но думаю, что не читаю правильно.
Вот что я получил до сих пор:
public static String toHex(byte[] bytes) {
BigInteger bi = new BigInteger(1, bytes);
return String.format("%0" + (bytes.length << 1) + "X", bi);
}
private void ProcessInitialPacket(){
int port = 1954;
System.out.println("Listening on port :"+port);
byte[] data = new byte[17];
byte[] ackPacket = new byte[2];
byte[] dataPacket= new byte[15];
try {
ServerSocket sSocket = new ServerSocket(port);
Socket cSocket = sSocket.accept();
DataInputStream dataIN = new DataInputStream(cSocket.getInputStream());
int packetSize=dataIN.read(data,0,data.length);
System.arraycopy(data, 0, ackPacket, 0, 2);
System.arraycopy(data,2,dataPacket,0,15);
System.out.println("Total packet size: "+ packetSize);
System.out.println("ACK PACKET : "+ toHex(ackPacket));
System.out.println("DATA PACKET: "+ toHex(dataPacket));
System.out.println("FULL PACKET: "+ toHex(data));
} catch (IOException e) {
e.printStackTrace();
}
}
выход:
-PARSER()--
-INITSESSION-- Прослушивание в порту:1954
Общий размер пакета: 17
ACK ПАКЕТ: 000F
ПАКЕТ ДАННЫХ: 333532383438303236323631393534
ПОЛНЫЙ ПАКЕТ: 000F333532383438303236323631393534
------ CLOSESESSION ------------
Теперь моя проблема:
здесь происходит то, что устройство отправляет [0x00][0x0F]xxxxxxxxxxxxxxx, где xxxxxxx - это его imei (пакет данных). Моя проблема в том, что в пакете данных слишком много 3-х, поэтому реальный действительный вывод
352848026261954
который вы получите, удалив 3. мой вопрос: может ли такое поведение происходить из моего кода или его части протокола? Я могу исправить это программно, но я хочу знать, что есть способ, которым код может вызвать эти дополнительные 3.
2 ответа
Вы смотрите на шестнадцатеричные значения ascii, которые должны быть декодированы как числа. Символ "0" в виде десятичного числа равен 48 или в виде шестнадцатеричного числа 0x30, до "9" - это 57 в виде десятичного числа или 0x39 в виде шестнадцатеричного числа.
Итак, последовательность байтов
33 35 32 38 34 38 30 32 36 32 36 31 39 35 34
является
"352848026261954"
как символы ASCII.
Я бы поменял твой код вот так
dataIN.readFully(data); // always read 17 bytes, not less
String text = new String(data, 0, 2, 13); // decodes 8-bit bytes as a String
long number = Long.parseLong(text);
Это цифровая кодировка ASCII. 0 передается как "0", что составляет 0x30. 1 передается как 0x31. И т.д. Итак, ваше понимание формата данных неверно.