!AIVDM,1,1,A,13u?etPv2;0n:dDPwUM1U1Cb069D,023*
我们知道消息内容就是13u?etPv2;0n:dDPwUM1U1Cb069D,这是一串ASCII码字符串,我们可以获取其对应的ASCII码数值。
但是在AIS的编码表不需要这么多符号,所以截取了其中一部分,如下表:
所以我们要想解码AIS报文,就需要把原始AIS报文中的ASCII码值转换为下面的值,然后再用下面的二进制拼接就可以了。
对比两张表我们很容易看出规律,就是AIS编码表0 ~ W(0 ~ 27)对应ASCII码中的48 ~ 87,(‘ ~ w) 对应96 ~ 119,所以我们可以初始化一个数组来转换:
int[] toSixbit = new int[256 * 256];
for (int chr = 0; chr < toSixbit.length; chr++) {
if (chr < 48 || chr > 119 || chr > 87 && chr < 96) {
toSixbit[chr] = -1;
} else if (chr < 0x60) {
toSixbit[chr] = chr - 48 & 0x3F;
} else {
toSixbit[chr] = chr - 56 & 0x3F;
}
}
接下来就只需要将转换后的数据的后六位拼接到一起就可以了,为了方便后面的取用,可以引入colt库使用BitVector来存储转换后的二进制数据:
<dependency>
<groupId>colt</groupId>
<artifactId>colt</artifactId>
<version>1.2.0</version>
</dependency>
获取后6位拼接
public static BitVector getBitVector(String ais) {
String[] split = ais.split(",");
String payload = split[5];
BitVector bitVector = new BitVector(payload.length() * 6);
for (int i = 0; i < payload.length(); i++) {
char asciiChar = payload.charAt(i);
int i1 = INT_TO_SIX_BIT[asciiChar];
for (int j = 0; j < 6; j++) {
if (((i1 >> (5 - j)) & 1) == 1) {
bitVector.set(i * 6 + j);
}
}
}
return bitVector;
}
成功拿到二进制数据后,按照协议解析就可以了,比如我们根据协议知道开头的AIS报文的第8位到37位表示MMSI,那就可以获取二进制串中的8~37位转为数值即可:
public static void main(String[] args) {
String ais = "!AIVDM,1,1,,A,13u?etPv2;0n:dDPwUM1U1Cb069D,0*23";
BitVector bitVector = getBitVector(ais);
long value = bitVector.getLongFromTo(8, 37);
long bits = 37 - 8 + 1;
long temp = Long.reverse(value << (int) (64L - bits));
System.out.println(temp);
}
可以看到解析MMSI结果为:265547250