In Order:
!AIVDM: The NMEA message type
1 Number of Sentences (some messages need more then one)
1 Sentence Number (1 unless it's a multi-sentence message)
The blank is the Sequential Message ID (for multi-sentence messages)
A The AIS Channel (A or B)
14eG;... The Encoded AIS Data
0* End of Data
7D NMEA Checksum
AIS Data Encoding
In the NMEA encoding for AIS - each ASCII character corresponds to 6 binary bits (unlike normal ASCII which uses 8 bits) so you need to step through each character and subtract 48 from the ASCII - then if it's still a decimal number > 40 subtract another 8 - then convert to binary: this guarantees a 6 bit number. Looking at our data (just the first few characters)14eG
1 = 000001
4 = 000100
e = 101101
G = 010111
and so on...
The complete data string decoded looks like this when strung back together.
000001 000100 101101 010111 011100 001010 010000 000000 000000 000000 110111 001000 110111 100001 101000 011100 011101 110010 011111 101011 110000 110101 010111 010000 000000 001000 011000 011011
Now you start grabbing sets of bits from this and converting to decimal. Here are the key pieces of information assuming the first character is a '1' (the message type):
MMSI Number - starting from bit 8 for 30 bits
= 010010110101011101110000101001 = 316005417
HDG - bit 128 for 9 bits
COG - bit 116 for 12 bits (and divide by 10)
SOG - bit 50 for 10 bits (and divide by 10)
Lat - bit 89 for 27 bits (a signed binary number, divide by 600000)
Lon - bit 61 for 28 bits (a signed binary number, divide by 600000)
import java.util.StringTokenizer;
public class AISDecoder {
public static void main(String[] args) {
String s = "!AIVDM,1,1,,A,14eG;o@034o8sd
StringTokenizer stok = new StringTokenizer(s, ",");
String tokens[] = new String[stok.countTokens()];
for (int i = 0;i< stok.countTokens()+5;i++) {
tokens[i] = stok.nextToken().trim();
}
//System.out.println(tokens[4]);
String msg = tokens[4];
String binarystr = "";
String full = "";
for (int i = 0; i < msg.length(); i++) {
int j = (int) msg.charAt(i);
//System.out.println(j);
j = j - 48;
//System.out.println(j);
if (j > 40) {
j = j - 8;
}
//System.out.println(j);
//System.out.println();
binarystr = Integer.toBinaryString(j);
int len = binarystr.length();
switch (len) {
case 1:
binarystr = "00000" + binarystr;
break;
case 2:
binarystr = "0000" + binarystr;
break;
case 3:
binarystr = "000" + binarystr;
break;
case 4:
binarystr = "00" + binarystr;
break;
case 5:
binarystr = "0" + binarystr;
break;
case 6:
binarystr = "" + binarystr;
break;
}
// System.out.println(binarystr);
// System.out.println("");
full = full.concat(binarystr).trim();
}
System.out.println(full);
//System.out.println(full.length());
String msgtype = full.substring(0, 6);
System.out.println(msgtype);
System.out.println(Integer.parseInt(msgtype,2));
String repind = full.substring(6, 8);
System.out.println(repind);
System.out.println(Integer.parseInt(repind,2));
String mmsi = full.substring(8, 38);
System.out.println("mm"+mmsi.length());
System.out.println(Integer.parseInt(mmsi,2));
String aisversion = full.substring(38,40);
System.out.println(Integer.parseInt(aisversion,2));
String imonum = full.substring(40,70);
System.out.println(Integer.parseInt(imonum,2));
String lon = full.substring(61, 88);
System.out.println("len" + lon.length());
int lonint = Integer.parseInt(lon,2);
float lonflo = (float)lonint/600000;
System.out.println(lonflo);
String lat = full.substring(89,117);
int latint = Integer.parseInt(lat,2);
float latflo = (float)latint/600000;
System.out.println(latflo);
// String callsign = full.substring(70,112);
// System.out.println(callsign.length());
//System.out.println(Double.parseDouble(ca));
// System.out.println(mmsi);
// System.out.println(Integer.parseInt(mmsi,2));
}
}
No comments:
Post a Comment