掌握进制转换:二进制转十六进制; 掌握BCD编码、Hex编码; 掌握各种位移、异或; 掌握常用反射; 掌握JObject的用法; 掌握快速ctrl+c、ctrl+v; 掌握Span<T>的基本用法; 掌握以上装逼技能,就可以开始搬砖了。 头标识 数据头 数据体 CRC校验码 尾标识 BeginFlag JT809Header JT809Bodies CRCCode EndFlag 5B - - - 5D
数据长度 报文序列号 业务数据类型 下级平台接入码 协议版本号标识 报文加密标识位 数据加密的密匙 发送消息的系统时间(2019版本) MsgLength MsgSN BusinessType MsgGNSSCENTERID Version EncryptFlag EncryptKey Time
根据对应业务数据类型:BusinessType
车牌号 车辆颜色 子业务类型标识 后续数据长度 子业务数据体 VehicleNo VehicleColor SubBusinessType DataLength JT809SubBodies
根据对应子业务数据类型 SubBusinessType 处理子业务数据体(JT809SubBodies)。
注意:数据内容(除去头和尾标识)进行转义判断
转义规则如下:
若数据内容中有出现字符 0x5b 的,需替换为字符 0x5a 紧跟字符 0x01; 若数据内容中有出现字符 0x5a 的,需替换为字符 0x5a 紧跟字符 0x02; 若数据内容中有出现字符 0x5d 的,需替换为字符 0x5e 紧跟字符 0x01; 若数据内容中有出现字符 0x5e 的,需替换为字符 0x5e 紧跟字符 0x02. 反转义的原因:确认JT809协议的TCP消息边界。
业务数据类型 BusinessType:从链路报警信息交互消息
子业务类型标识 SubBusinessType:报警督办请求消息
JT809Package jT809Package = new JT809Package();jT809Package.Header = new JT809Header{ MsgSN = 1666, EncryptKey = 9999, EncryptFlag= JT809Header_Encrypt.None, Version = new JT809Header_Version(1, 0, 0), BusinessType = JT809BusinessType.从链路报警信息交互消息.ToUInt16Value(), MsgGNSSCENTERID = 20180920,};JT809_0x9400 bodies = new JT809_0x9400{ VehicleNo="粤A12345", VehicleColor= JT809VehicleColorType.黄色, SubBusinessType= JT809SubBusinessType.报警督办请求消息.ToUInt16Value(),};JT809_0x9400_0x9401 jT809_0x9400_0x9401 = new JT809_0x9400_0x9401{ WarnSrc = JT809WarnSrc.车载终端, WarnType = JT809WarnType.疲劳驾驶报警.ToUInt16Value(), WarnTime = DateTime.Parse("2018-09-27 10:24:00"), SupervisionID = "123FFAA1", SupervisionEndTime = DateTime.Parse("2018-09-27 11:24:00"), SupervisionLevel = 3, Supervisor = "smallchi", SupervisorTel = "12345678901", SupervisorEmail = "123456@qq.com"};bodies.SubBodies = jT809_0x9400_0x9401;jT809Package.Bodies = bodies;byte[] data = JT809Serializer.Serialize(jT809Package);string hex = data.ToHexString();// 输出结果Hex:// 5B 00 00 00 92 00 00 06 82 94 00 01 33 EF B8 01 00 00 00 00 00 27 0F D4 C1 41 31 32 33 34 35 00 00 00 00 00 00 00 00 00 00 00 00 00 02 94 01 00 00 00 5C 01 00 02 00 00 00 00 5A 01 AC 3F 40 12 3F FA A1 00 00 00 00 5A 01 AC 4D 50 03 73 6D 61 6C 6C 63 68 69 00 00 00 00 00 00 00 00 31 32 33 34 35 36 37 38 39 30 31 00 00 00 00 00 00 00 00 00 31 32 33 34 35 36 40 71 71 2E 63 6F 6D 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 BA D8 5D
1.原包:5B 00 00 00 92 00 00 06 82 94 00 01 33 EF B8 01 00 00 00 00 00 27 0F D4 C1 41 31 32 33 34 35 00 00 00 00 00 00 00 00 00 00 00 00 00 02 94 01 00 00 00 5C 01 00 02 00 00 00 00 (5A 01) AC 3F 40 12 3F FA A1 00 00 00 00 (5A 01) AC 4D 50 03 73 6D 61 6C 6C 63 68 69 00 00 00 00 00 00 00 00 31 32 33 34 35 36 37 38 39 30 31 00 00 00 00 00 00 00 00 00 31 32 33 34 35 36 40 71 71 2E 63 6F 6D 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 BA D8 5D2.进行反转义5A 01 ->5B5A 02 ->5A5E 01 ->5D5E 02 ->5E反转义后5B 00 00 00 92 00 00 06 82 94 00 01 33 EF B8 01 00 00 00 00 00 27 0F D4 C1 41 31 32 33 34 35 00 00 00 00 00 00 00 00 00 00 00 00 00 02 94 01 00 00 00 5C 01 00 02 00 00 00 00 5B AC 3F 40 12 3F FA A1 00 00 00 00 5B AC 4D 50 03 73 6D 61 6C 6C 63 68 69 00 00 00 00 00 00 00 00 31 32 33 34 35 36 37 38 39 30 31 00 00 00 00 00 00 00 00 00 31 32 33 34 35 36 40 71 71 2E 63 6F 6D 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 BA D8 5D3.拆解5B --头标识00 00 00 92 --数据头->数据长度00 00 06 82 --数据头->报文序列号94 00 --数据头->业务数据类型01 33 EF B8 --数据头->下级平台接入码,上级平台给下级平台分配唯一标识码01 00 00 --数据头->协议版本号标识00 --数据头->报文加密标识位00 00 27 0F --数据头->数据加密的密匙D4 C1 41 31 32 33 34 35 00 00 00 00 00 00 00 00 00 00 00 00 00 --数据体->车牌号02 --数据体->车辆颜色94 01 --数据体->子业务类型标识00 00 00 5C --数据体->后续数据长度01 --子数据体->报警信息来源00 02 --子数据体->报警类型00 00 00 00 5B AC 3F 40 --子数据体->报警时间UTCDateTime12 3F FA A1 --子数据体->报警督办ID00 00 00 00 5B AC 4D 50 --子数据体->督办截止时间03 --子数据体->督办级别73 6D 61 6C 6C 63 68 69 00 00 00 00 00 00 00 00 --子数据体->督办人31 32 33 34 35 36 37 38 39 30 31 00 00 00 00 00 00 00 00 00 --子数据体->督办联系电话31 32 33 34 35 36 40 71 71 2E 63 6F 6D 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 --子数据体->督办联系电子邮件BA D8 --CRC校验码5D --尾标识
//1.转成byte数组var bytes = "5B 00 00 00 92 00 00 06 82 94 00 01 33 EF B8 01 00 00 00 00 00 27 0F D4 C1 41 31 32 33 34 35 00 00 00 00 00 00 00 00 00 00 00 00 00 02 94 01 00 00 00 5C 01 00 02 00 00 00 00 5A 01 AC 3F 40 12 3F FA A1 00 00 00 00 5A 01 AC 4D 50 03 73 6D 61 6C 6C 63 68 69 00 00 00 00 00 00 00 00 31 32 33 34 35 36 37 38 39 30 31 00 00 00 00 00 00 00 00 00 31 32 33 34 35 36 40 71 71 2E 63 6F 6D 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 BA D8 5D".ToHexBytes();//2.将数组反序列化JT809Package jT809Package = JT809Serializer.Deserialize(bytes);//3.数据包头Assert.Equal((uint)146, jT809Package.Header.MsgLength);Assert.Equal((uint)1666, jT809Package.Header.MsgSN);Assert.Equal((uint)9999, jT809Package.Header.EncryptKey);Assert.Equal(JT809Header_Encrypt.None, jT809Package.Header.EncryptFlag);Assert.Equal((uint)20180920, jT809Package.Header.MsgGNSSCENTERID);Assert.Equal(JT809BusinessType.从链路报警信息交互消息, (JT809BusinessType)jT809Package.Header.BusinessType);Assert.Equal(new JT809Header_Version(1, 0, 0).ToString(), jT809Package.Header.Version.ToString());//4.数据包体JT809_0x9400 jT809_0X400 = (JT809_0x9400)jT809Package.Bodies;Assert.Equal("粤A12345", jT809_0X400.VehicleNo);Assert.Equal(JT809VehicleColorType.黄色, jT809_0X400.VehicleColor);Assert.Equal(JT809SubBusinessType.报警督办请求消息, (JT809SubBusinessType)jT809_0X400.SubBusinessType);Assert.Equal((uint)92, jT809_0X400.DataLength);//5.子数据包体JT809_0x9400_0x9401 jT809_0x9400_0x9401 = (JT809_0x9400_0x9401)jT809_0X400.JT809SubBodies;Assert.Equal(JT809WarnSrc.车载终端, jT809_0x9400_0x9401.WarnSrc);Assert.Equal(JT809WarnType.疲劳驾驶报警, (JT809WarnType)jT809_0x9400_0x9401.WarnType);Assert.Equal(DateTime.Parse("2018-09-27 10:24:00"), jT809_0x9400_0x9401.WarnTime);Assert.Equal("123FFAA1", jT809_0x9400_0x9401.SupervisionID);Assert.Equal(DateTime.Parse("2018-09-27 11:24:00"), jT809_0x9400_0x9401.SupervisionEndTime);Assert.Equal(3, jT809_0x9400_0x9401.SupervisionLevel);Assert.Equal("smallchi", jT809_0x9400_0x9401.Supervisor);Assert.Equal("12345678901", jT809_0x9400_0x9401.SupervisorTel);Assert.Equal("123456@qq.com", jT809_0x9400_0x9401.SupervisorEmail);
// 根据业务类型创建对应包JT809Package jT809Package = JT809BusinessType.从链路报警信息交互消息.Create_从链路报警信息交互消息( new JT809Header { MsgSN = 1666, EncryptKey = 9999, EncryptFlag = JT809Header_Encrypt.None, Version = new JT809Header_Version(1, 0, 0), MsgGNSSCENTERID = 20180920, }, new JT809_0x9400 { VehicleNo = "粤A12345", VehicleColor = JT809VehicleColorType.黄色, SubBusinessType = JT809SubBusinessType.报警督办请求消息.ToUInt16Value(), SubBodies = JT809SubBusinessType.报警督办请求消息.Create_报警督办请求消息( new JT809_0x9400_0x9401 { WarnSrc = JT809WarnSrc.车载终端, WarnType = JT809WarnType.疲劳驾驶报警.ToUInt16Value(), WarnTime = DateTime.Parse("2018-09-27 10:24:00"), SupervisionID = "123FFAA1", SupervisionEndTime = DateTime.Parse("2018-09-27 11:24:00"), SupervisionLevel = 3, Supervisor = "smallchi", SupervisorTel = "12345678901", SupervisorEmail = "123456@qq.com" }) });var hex = JT809Serializer.Serialize(jT809Package).ToHexString();
static void Main(string[] args){ class JT809GlobalConfig: GlobalConfigBase { public override JT809EncryptOptions EncryptOptions { get; set; }= new JT809EncryptOptions() { IA1 = 20000000, IC1 = 20000000, M1 = 30000000 }; public override string ConfigId => "JT809GlobalConfig"; } JT809Serializer JT809Serializer = new JT809Serializer(new JT809GlobalConfig()); // todo:}
如何在项目中同时使用809的2011和2019版本
IServiceCollection serviceDescriptors = new ServiceCollection();serviceDescriptors.AddJT809Configure(new JT809_2011_Config());serviceDescriptors.AddJT809Configure(new JT809_2019_Config());JT809Serializer JT809_2011_Serializer= ServiceProvider.GetRequiredService<JT809_2011_Config>().GetSerializer();JT809Serializer JT809_2019_Serializer = ServiceProvider.GetRequiredService<JT809_2019_Config>().GetSerializer();public class JT809_2011_Config: JT809GlobalConfigBase{ public override string ConfigId => "JT809_2011";}public class JT809_2019_Config : JT809GlobalConfigBase{ public override string ConfigId => "JT809_2019"; public JT809_2019_Config() { Version = JT809Version.JTT2019; }}
如何在项目中同时使用808的2019版本和809的2019版本
IServiceCollection serviceDescriptors = new ServiceCollection();serviceDescriptors.AddJT808Configure(new JT808_2019_Config());serviceDescriptors.AddJT809Configure(new JT809_2019_Config());JT808Serializer JT808_2019_Serializer = ServiceProvider.GetRequiredService<JT808_2019_Config>().GetSerializer();JT809Serializer JT809_2019_Serializer = ServiceProvider.GetRequiredService<JT809_2019_Config>().GetSerializer();public class JT808_2019_Config : GlobalConfigBase{ public override string ConfigId { get; protected set; } public JT808_2019_Config(string configId = "jt808_2019") { ConfigId = configId; }}public class JT809_2019_Config : JT809GlobalConfigBase{ public override string ConfigId => "JT809_2019"; public JT809_2019_Config() { Version = JT809Version.JTT2019; }}
Package Name Version Pre Version Downloads Install-Package JT809 Install-Package JT809.Protocol.Extensions.JT1078
使用BenchmarkDotNet性能测试报告(只是玩玩,不能当真) BenchmarkDotNet v0.14.0, Windows 11 (10.0.22631.4460/23H2/2023Update/SunValley3)Intel Core i7-8700K CPU 3.70GHz (Coffee Lake), 1 CPU, 12 logical and 6 physical cores.NET SDK 9.0.100 [Host] : .NET 9.0.0 (9.0.24.52809), X64 RyuJIT AVX2 Job-IFEFGZ : .NET 9.0.0 (9.0.24.52809), X64 RyuJIT AVX2Platform =AnyCpuServer =FalseToolchain =.NET 9.0 Method N Mean Error StdDev Gen0 Allocated JT809_0x9400_0x9401_Package_Deserialize 100 363.7 μs 6.42 μs 6.00 μs 16.6016 103.13 KB JT809_0x9400_0x9401_Package_Serialize 100 147.9 μs 1.64 μs 1.45 μs 14.6484 90.63 KB JT809_0x9400_0x9401_Package_Deserialize 10000 35,620.3 μs 470.76 μs 440.35 μs 1642.8571 10312.58 KB JT809_0x9400_0x9401_Package_Serialize 10000 14,931.2 μs 217.72 μs 203.65 μs 1468.7500 9062.52 KB JT809_0x9400_0x9401_Package_Deserialize 100000 372,631.6 μs 7,066.99 μs 6,610.47 μs 16000.0000 103126.05 KB JT809_0x9400_0x9401_Package_Serialize 100000 151,873.4 μs 1,104.01 μs 921.90 μs 14750.0000 90625.26 KB
2019版8.3.3.2.5 新增的 交换车辆行驶路线信息消息,无对应消息子id类型,暂未对接
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1001 √ √ 主链路登录请求消息 2 0x1002 √ √ 主链路登录应答消息 3 0x1003 √ √ 主链路注销请求消息 4 0x1004 √ √ 主链路注销应答消息 5 0x1005 √ √ 主链路连接保持请求消息 6 0x1006 √ √ 主链路连接保持应答消息 7 0x1007 √ √ 主链路断开通知消息 8 0x1008 √ √ 下级平台主动关闭链路通知消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9001 √ √ 从链路连接请求消息 2 0x9002 √ √ 从链路连接应答消息 3 0x9003 √ √ 从链路注销请求消息 4 0x9004 √ √ 从链路注销应答消息 5 0x9005 √ √ 从链路连接保持请求消息 6 0x9006 √ √ 从链路连接保持应答消息 7 0x9007 √ √ 从链路断开通知消息 8 0x9008 √ √ 上级平台主动关闭链路通知消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9101 √ √ 接收定位信息数量通知消息 1 0x9102 √ √ 平台链路连接情况与车辆定位消息传输情况上报请求消息 1 0x9103 √ √ 下发平台间消息序列号通知消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1200 √ √ 主链路动态信息交换消息 2 0x1201 √ √ 上传车辆注册信息(809补充协议文档) 3 0x1202 √ √ 实时上传车辆定位信息 4 0x1203 √ √ 车辆定位信息自动补报 5 0x1205 √ √ 启动车辆定位信息交换应答消息 6 0x1206 √ √ 结束车辆定位信息交换应答消息 7 0x1207 √ √ 申请交换指定车辆定位信息请求消息 8 0x1208 √ √ 取消交换指定车辆定位信息请求 9 0x1209 √ √ 补发车辆定位信息请求 10 0x120A √ √ 上报车辆驾驶员身份识别信息应答 11 0x120B √ √ 上报车辆电子运单应答 12 0x120C √ √ 主动上报驾驶员身份信息 13 0x120D √ √ 主动上报车辆电子运单信息 14 0x120E √ √ 主动上报车辆行驶路线信息(809-2019)
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9200 √ √ 从链路动态信息交换消息 1 0x9201 √ √(0x1201) 车辆注册信息应答消息 2 0x9202 √ √(0x1202) 交换车辆定位信息消息(809补充协议文档) 3 0x9203 √ √(0x1203) 车辆定位信息交换补发消息 4 0x9204 √ √ 交换车辆静态信息消息 5 0x9205 √ √ 启动车辆定位信息交换请求消息 6 0x9206 √ √ 结束车辆定位信息交换请求消息 7 0x9207 √ √ 申请交换指定车辆定位信息应答消息 8 0x9208 √ √ 取消申请交换指定车辆定位信息应答消息 9 0x9209 √ √ 补发车辆定位信息应答消息 10 0x920A √ √ 上报驾驶员身份识别信息请求消息 11 0x920B √ √ 上报车辆电子运单请求消息 12 0x920C √ √ 上报车辆车辆行驶路线请求(809-2019) 13 0x920D √ √ 车辆行驶线路请求应答(809-2019)
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1300 √ √ 主链路平台间信息交互消息 2 0x1301 √ √ 平台查岗应答消息(809补充协议文档) 3 0x1302 √ √ 下发平台间报文应答消息(809补充协议文档) 4 0x1303 √ √ 上传平台间消息补传请求消息(809-2019)
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9300 √ √ 从链路平台间信息交互消息 2 0x9301 √ √ 平台查岗请求(809补充协议文档) 3 0x9302 √ √ 下发平台间报文请求(809补充协议文档) 4 0x9303 √ √ 下发平台间消息补传请求消息(809-2019)
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1400 √ √ 主链路平台间信息交互消息 2 0x1401 √ √ 报警督办应答消息 3 0x1402 √ √ 上报报警信息消息 4 0x1403 √ √ 主动上报报警处理结果信息(809补充协议文档)
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9400 √ √ 主链路平台间信息交互消息 2 0x9401 √ √ 报警督办请求 3 0x9402 √ √ 报警预警 4 0x9403 √ √ 实时交换报警信息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1500 √ √ 主链路车辆监管消息 2 0x1501 √ √ 车辆单向监听应答 3 0x1502 √ √ 车辆拍照应答 4 0x1503 √ √ 下发车辆报文应答 5 0x1504 √ √ 上报车辆行驶记录应答(809补充协议文档) 6 0x1505 √ √ 车辆应急接入监管平台应答消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9500 √ √ 从链路车辆监管消息 2 0x9501 √ √ 车辆单向监听请求 3 0x9502 √ √ 车辆拍照请求 4 0x9503 √ √ 下发车辆报文请求 5 0x9504 √ √ 上报车辆行驶记录请求 6 0x9505 √ √ 车辆应急接入监管平台请求消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1600 √ √ 主链路静态信息交换消息 2 0x1601 √ √ 补报车辆静态信息应答 3 0x1602 √ √ 补报车辆行驶路线信息应答消息
序号 消息ID 完成情况 测试情况 消息体名称 1 0x9600 √ √ 从链路静态信息交换消息 2 0x9601 √ √ 补报车辆静态信息应答
序号 消息ID 完成情况 测试情况 消息体名称 1 0x1700 √ √ 主链路时效口令交互消息 2 0x1700_0x1701 √ √ 时效口令上报消息(有疑问:数据体有问题) 3 0x1700_0x1702 √ √ 时效口令请求消息 4 0x1800 √ √ 主链路实时音视频交互消息 5 0x1800_0x1801 √ √ 实时音视频请求应答消息 6 0x1800_0x1802 √ √ 主动请求停止实时音视频传输应答消息 7 0x1900 √ √ 主链路远程录像检索 8 0x1900_0x1901 √ √ 主动上传音视频资源目录信息消息 9 0x1900_0x1902 √ √ 查询音视频资源目录应答消息 10 0x1A00 √ √ 主链路远程录像回放交互消息 11 0x1A00_0x1A01 √ √ 远程录像回放请求应答消息 12 0x1A00_0x1A02 √ √ 远程录像回放控制应答消息 13 0x1B00 √ √ 主链路远程录像下载交互消息 14 0x1B00_0x1B01 √ √ 远程录像下载请求应答消息 15 0x1B00_0x1B02 √ √ 远程录像下载通知消息 16 0x1B00_0x1B03 √ √ 远程录像下载控制应答消息
序号 消息ID 完成情况 测试情况 消息体名称 17 0x9700 √ √ 从链路时效口令交互消息 18 0x9700_0x9702 √ √ 时效口令请求应答消息(有疑问:应该有应答结果) 19 0x9800 √ √ 从链路实时音视频交互信息 20 0x9800_0x9801 √ √ 实时音视频请求消息 21 0x9800_0x9802 √ √ 主动请求停止实时音视频传输消息 22 0x9900 √ √ 从链路远程录像检索交互消息 23 0x9900_0x9901 √ √ 主动上传音视频资源目录信息应答消息 24 0x9900_0x9902 √ √ 查询音视频资源目录请求消息 25 0x9A00 √ √ 从链路远程录像回放交互消息 26 0x9A00_0x9A01 √ √ 远程录像回放请求消息 27 0x9A00_0x9A02 √ √ 远程录像回放控制消息 28 0x9B00 √ √ 从链路远程录像下载交互消息 29 0x9B00_0x9B01 √ √ 远程录像下载请求消息 30 0x9B00_0x9B02 √ √ 远程录像下载完成通知应答消息 31 0x9B00_0x9B03 √ √ 远程录像下载控制消息
IServiceCollection serviceDescriptors1 = new ServiceCollection();serviceDescriptors1.AddJT809Configure() .AddJT1078Configure();