PosApi.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Runtime.InteropServices;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. using XdCxRhDW.Repostory.Model;
  8. namespace XdCxRhDW.Core.Api
  9. {
  10. /// <summary>
  11. ///TODO 定位算法接口.在这里调用罗博士的算法库
  12. /// </summary>
  13. public static class PosApi
  14. {
  15. #region cpp dll Interop
  16. //一星一地测向带参定位
  17. private const string XDCX = @"AddIns\DLL_DTO_DOA_DW.dll";
  18. //两星一地带参定位
  19. private const string X2D1 = @"AddIns\DLL_SC_2X1D_DW.dll";
  20. //三星双时差带参、三星双时差无参、三星双频差带参、双星时频差带参、两星一地无参定位及时差线
  21. private const string OtherPos = @"AddIns\DLL_11J_DW.dll";
  22. [DllImport(XDCX, EntryPoint = "XD_CX_DW", CallingConvention = CallingConvention.Cdecl)]//一星一地测向带参
  23. private extern static void X1D1_Pos20240305_Core(double[] mainSat, double[] satStation, double[] cdbStation, double[] cxStation, double[] refStation, double[] zone, double theta, double tarDto, double refDto, double[] res);
  24. [DllImport(X2D1, EntryPoint = "SC_2X1D_DW", CallingConvention = CallingConvention.Cdecl)]//两星一地带参
  25. private extern static void X2D1_Pos20240305_Core(double[] mainSat, double[] adjaSat, double[] cdbStation, double[] satStation1, double[] satStation2, double[] satStation3, double[] satStation4,
  26. double[] satStation5, double[] refStation, double[] zone, double tarSxDto, double tarXdDto, double samp_main_dto, double samp_neigh_dto, double[] res);
  27. [DllImport(OtherPos, EntryPoint = "XingDi_2X1D_DW_NoRef", CallingConvention = CallingConvention.Cdecl)]//两星一地无参
  28. private extern static void X2D1_PosNoRef20240305_Core(double[] mainSat, double[] adjaSat, double[] cdbStation, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  29. , double[] zone, double tarSxDto, double tarXdDto, double[] res);
  30. [DllImport(OtherPos, EntryPoint = "SanXing_DW", CallingConvention = CallingConvention.Cdecl)]//三星双时差带参
  31. private extern static void X3_Pos20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  32. , double[] mainSatRefStation, double[] adjaSat1RefStation, double[] adjaSat2RefStation, double[] refStation, double[] zone, double tarDto1, double tarDto2, double refDto1, double refDto2, double[] res);
  33. [DllImport(OtherPos, EntryPoint = "SanXing_DW_NoRef", CallingConvention = CallingConvention.Cdecl)]//三星双时差无参
  34. private extern static void X3_PosNoRef20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  35. , double[] zone, double tarDto1, double tarDto2, double[] res);
  36. [DllImport(OtherPos, EntryPoint = "TriStar_2DFO_DW", CallingConvention = CallingConvention.Cdecl)]//三星双频差带参
  37. private extern static void X3_PosTwoDfo20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  38. , double[] refStation, double[] zone, double tarDfo1, double tarDfo2, double refDfo1, double refDfo2, double fu1, double fd1, double fu2, double fd2, double[] res);
  39. [DllImport(OtherPos, EntryPoint = "TwoStar_DTFO_DW", CallingConvention = CallingConvention.Cdecl)]//双星时频差带参
  40. private extern static void X2_Pos20240305_Core(double[] mainSat, double[] adjaSat, double[] mainSatTarStation, double[] adjaSatTarStation, double[] refStation, double[] zone
  41. , double tarDto, double tarDfo, double refDto, double refDfo, double fu1, double fd1, double fu2, double fd2, double[] res);
  42. #endregion
  43. /// <summary>
  44. /// 一星一地带参,返返回经度纬度高度及镜像点,数组长度为6
  45. /// </summary>
  46. /// <param name="cgRes">参估结果</param>
  47. /// <param name="sRes">站点信息</param>
  48. /// <param name="cxRes">测向结果</param>
  49. /// <returns></returns>
  50. public static double[] X1D1_Pos(CgRes cgRes, StationRes sRes, CxRes cxRes)
  51. {
  52. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  53. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  54. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  55. double[] cxStation = new double[3] { sRes.CxLon.Value, sRes.CxLat.Value, 0 };
  56. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  57. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  58. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  59. double theta = cxRes.Fx;//单位°
  60. double[] res = new double[6];
  61. var ybDto1 = cgRes.YbMainDto.Value / 1e6;
  62. X1D1_Pos20240305_Core(mainSat, satStation, cdbStation, cxStation, refStation, zone, theta, dtoCdb, ybDto1, res);
  63. return ConvertToGeoPoint(res);
  64. }
  65. /// <summary>
  66. /// 两星一地带参,返回经度纬度高度及镜像点,数组长度为6
  67. /// </summary>
  68. /// <param name="cgRes">参估结果</param>
  69. /// <param name="sRes">站点信息</param>
  70. /// <returns></returns>
  71. public static double[] X2D1_Pos(CgRes cgRes, StationRes sRes)
  72. {
  73. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  74. double[] adjaSat = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  75. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  76. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  77. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  78. double dto1 = cgRes.Dto1.Value / 1e6;
  79. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  80. double ybDto1 = cgRes.YbMainDto.Value / 1e6;
  81. double ybDto2 = cgRes.YbAdja1Dto.Value / 1e6;
  82. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  83. double[] res = new double[6];
  84. X2D1_Pos20240305_Core(mainSat, adjaSat, cdbStation, satStation, satStation, satStation, satStation, satStation, refStation, zone, dto1, dtoCdb, ybDto1, ybDto2, res);
  85. return ConvertToGeoPoint(res);
  86. }
  87. /// <summary>
  88. /// 两星一地无参,返回经度纬度高度及镜像点,数组长度为6
  89. /// </summary>
  90. /// <param name="cgRes">参估结果</param>
  91. /// <param name="sRes">站点信息</param>
  92. /// <returns></returns>
  93. public static double[] X2D1_PosNoRef(CgRes cgRes, StationRes sRes)
  94. {
  95. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  96. double[] adjaSat = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  97. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  98. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  99. double dto1 = cgRes.Dto1.Value / 1e6;
  100. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  101. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  102. double[] res = new double[6];
  103. X2D1_PosNoRef20240305_Core(mainSat, adjaSat, cdbStation, satStation, satStation, satStation, zone, dto1, dtoCdb, res);
  104. return ConvertToGeoPoint(res);
  105. }
  106. /// <summary>
  107. /// 融合定位带参,返回经度纬度高度及镜像点,数组长度为6
  108. /// </summary>
  109. /// <param name="cgRes">参估结果</param>
  110. /// <param name="sRes">站点信息</param>
  111. /// <param name="cxRes">测向结果</param>
  112. /// <returns></returns>
  113. public static double[] RH_Pos(CgRes cgRes, StationRes sRes, CxRes cxRes)
  114. {
  115. var res1 = X1D1_Pos(cgRes, sRes, cxRes);
  116. var res2 = X2D1_Pos(cgRes, sRes);
  117. double[] res = new double[] { 999, 999, 0, 999, 999, 0 };
  118. var p1 = res1.Take(3).ToArray();
  119. var p2 = res1.Skip(3).ToArray();
  120. var p3 = res2.Take(3).ToArray();
  121. var p4 = res2.Skip(3).ToArray();
  122. if (IsGeoPoint(p1) && IsGeoPoint(p2) && IsGeoPoint(p3) && IsGeoPoint(p4))
  123. {
  124. res = new double[6] {
  125. (p1[0] + p3[0]) / 2 + 0.003,
  126. (p1[1] + p3[1]) / 2 - 0.002,
  127. 0,
  128. (p2[0] + p4[0]) / 2 + 0.003,
  129. (p2[1] + p4[1]) / 2 - 0.002,
  130. 0,
  131. };
  132. }
  133. else if (IsGeoPoint(p1) && IsGeoPoint(p3))
  134. {
  135. res = new double[6] {
  136. (p1[0] + p3[0]) / 2 + 0.003,
  137. (p1[1] + p3[1]) / 2 - 0.002,
  138. 0,
  139. 999,
  140. 999,
  141. 0,
  142. };
  143. }
  144. else if (IsGeoPoint(p1) && IsGeoPoint(p2))
  145. {
  146. res = new double[6]
  147. {
  148. p1[0] + 0.003,
  149. p1[1] - 0.002,
  150. 0,
  151. p2[0] + 0.003,
  152. p2[1] - 0.002,
  153. 0,
  154. };
  155. }
  156. else if (IsGeoPoint(p3) && IsGeoPoint(p4))
  157. {
  158. res = new double[6]
  159. {
  160. p3[0] + 0.003,
  161. p3[1] - 0.002,
  162. 0,
  163. p4[0] + 0.003,
  164. p4[1] - 0.002,
  165. 0,
  166. };
  167. }
  168. return res;
  169. }
  170. /// <summary>
  171. /// 三星双时差带参,返回经度纬度高度及镜像点,数组长度为6
  172. /// </summary>
  173. /// <param name="cgRes">参估结果</param>
  174. /// <param name="sRes">站点信息</param>
  175. /// <returns></returns>
  176. public static double[] X3_Pos(CgRes cgRes, StationRes sRes)
  177. {
  178. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  179. double[] adjaSat1 = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  180. double[] adjaSat2 = new double[3] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value };
  181. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  182. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  183. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  184. var tarDto1 = cgRes.Dto1.Value / 1e6;
  185. var tarDto2 = cgRes.Dto2.Value / 1e6;
  186. var refDto1 = (cgRes.YbMainDto.Value - cgRes.YbAdja1Dto.Value) / 1e6;
  187. var refDto2 = (cgRes.YbMainDto.Value - cgRes.YbAdja2Dto.Value) / 1e6;
  188. double[] res = new double[6];
  189. X3_Pos20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation, satStation,
  190. satStation, satStation, refStation, zone, tarDto1, tarDto2, refDto1, refDto2, res);
  191. return ConvertToGeoPoint(res);
  192. }
  193. /// <summary>
  194. /// 三星双时差无参,返回经度纬度高度及镜像点,数组长度为6
  195. /// </summary>
  196. /// <param name="cgRes">参估结果</param>
  197. /// <param name="sRes">站点信息</param>
  198. /// <returns></returns>
  199. public static double[] X3_PosNoRef(CgRes cgRes, StationRes sRes)
  200. {
  201. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  202. double[] adjaSat1 = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  203. double[] adjaSat2 = new double[3] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value };
  204. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  205. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  206. var tarDto1 = cgRes.Dto1.Value / 1e6;
  207. var tarDto2 = cgRes.Dto2.Value / 1e6;
  208. double[] res = new double[6];
  209. X3_PosNoRef20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation, zone, tarDto1, tarDto2, res);
  210. return ConvertToGeoPoint(res);
  211. }
  212. /// <summary>
  213. /// 三星双频差带参,返回经度纬度高度及镜像点,数组长度为6
  214. /// </summary>
  215. /// <param name="cgRes">参估结果</param>
  216. /// <param name="sRes">站点信息</param>
  217. /// <returns></returns>
  218. public static double[] X3_PosTwoDfo(CgRes cgRes, StationRes sRes)
  219. {
  220. double[] mainSat = new double[6] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value, cgRes.MainVx.Value, cgRes.MainVy.Value, cgRes.MainVz.Value };
  221. double[] adjaSat1 = new double[6] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value, cgRes.Adja1Vx.Value, cgRes.Adja1Vy.Value, cgRes.Adja1Vz.Value };
  222. double[] adjaSat2 = new double[6] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value, cgRes.Adja2Vx.Value, cgRes.Adja2Vy.Value, cgRes.Adja2Vz.Value };
  223. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  224. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  225. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  226. var tarDfo1 = cgRes.Dfo1.Value;
  227. var tarDfo2 = cgRes.Dfo2.Value;
  228. var refDfo1 = cgRes.YbMainDfo.Value - cgRes.YbAdja1Dfo.Value;
  229. var refDfo2 = cgRes.YbMainDfo.Value - cgRes.YbAdja2Dfo.Value;
  230. double fu1 = cgRes.TarFreqUp.Value;
  231. double fd1 = cgRes.TarFreqDown.Value;
  232. double fu2 = cgRes.RefFreqUp.Value;
  233. double fd2 = cgRes.RefFreqDown.Value;
  234. double[] res = new double[6];
  235. X3_PosTwoDfo20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation,
  236. refStation, zone, tarDfo1, tarDfo2, refDfo1, refDfo2, fu1, fd1, fu2, fd2, res);
  237. return ConvertToGeoPoint(res);
  238. }
  239. /// <summary>
  240. /// 双星时频差带参,返回经度纬度高度及镜像点,数组长度为6
  241. /// </summary>
  242. /// <param name="cgRes">参估结果</param>
  243. /// <param name="sRes">站点信息</param>
  244. /// <returns></returns>
  245. public static double[] X2_PosDtoDfo(CgRes cgRes, StationRes sRes)
  246. {
  247. double[] mainSat = new double[6] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value, cgRes.MainVx.Value, cgRes.MainVy.Value, cgRes.MainVz.Value };
  248. double[] adjaSat = new double[6] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value, cgRes.Adja1Vx.Value, cgRes.Adja1Vy.Value, cgRes.Adja1Vz.Value };
  249. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  250. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  251. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  252. var tarDto = cgRes.Dto1.Value / 1e6;
  253. var tarDfo = cgRes.Dfo1.Value;
  254. var refDto = (cgRes.YbMainDto.Value - cgRes.YbAdja1Dto.Value) / 1e6;
  255. var refDfo = cgRes.YbMainDfo.Value - cgRes.YbAdja1Dfo.Value;
  256. double fu1 = cgRes.TarFreqUp.Value;
  257. double fd1 = cgRes.TarFreqDown.Value;
  258. double fu2 = cgRes.RefFreqUp.Value;
  259. double fd2 = cgRes.RefFreqDown.Value;
  260. double[] res = new double[6];
  261. X2_Pos20240305_Core(mainSat, adjaSat, satStation, satStation, refStation, zone, tarDto, tarDfo, refDto, refDfo, fu1, fd1, fu2, fd2, res);
  262. return ConvertToGeoPoint(res);
  263. }
  264. private static double[] ConvertToGeoPoint(double[] res)
  265. {
  266. if (res == null || res.Length == 0)
  267. {
  268. return new double[6] { 999, 999, 0, 999, 999, 0 };
  269. }
  270. else if (res.Length != 3 && res.Length != 6)
  271. {
  272. throw new Exception("定位结果解析异常,长度不是3或6");
  273. }
  274. var list = res.Select(p => Math.Round(p, 4)).ToArray();
  275. if (list.Length == 3)
  276. {
  277. list = list.Concat(new double[3] { 999, 999, 0 }).ToArray();
  278. }
  279. list[2] = list[5] = 0;//高度
  280. if (double.IsNaN(list[3]) || double.IsNaN(list[4]))
  281. {
  282. list[3] = list[4] = 999;
  283. }
  284. if (double.IsNaN(list[0]) || double.IsNaN(list[1]))
  285. {
  286. list[0] = list[1] = 999;
  287. }
  288. if (list[0] < -180 || list[0] > 180)
  289. {
  290. list[0] = list[1] = 999;
  291. }
  292. else if (list[1] < -90 || list[1] > 90)
  293. {
  294. list[0] = list[1] = 999;
  295. }
  296. if (list[3] < -180 || list[3] > 180)
  297. {
  298. list[3] = list[4] = 999;
  299. }
  300. else if (list[4] < -90 || list[4] > 90)
  301. {
  302. list[3] = list[4] = 999;
  303. }
  304. return list;
  305. }
  306. private static bool IsGeoPoint(double[] res)
  307. {
  308. if (res.Length != 3) return false;
  309. if (res[0] < -180 || res[0] > 180)
  310. {
  311. return false;
  312. }
  313. else if (res[1] < -90 || res[1] > 90)
  314. {
  315. return false;
  316. }
  317. return true;
  318. }
  319. }
  320. }