AircraftSJ.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453
  1. using KYFramework;
  2. using Model;
  3. using MongoDB.Bson;
  4. using MuShiApp;
  5. using Org.BouncyCastle.Asn1.Pkcs;
  6. using SimulationCommon;
  7. using SimulationSingleServer.Utils;
  8. using Unity.Mathematics;
  9. using Point = SimulationCommon.Point;
  10. using Random = System.Random;
  11. namespace SimulationServer;
  12. public class AircraftSJ : AircraftEntity
  13. {
  14. public bool IsOver;
  15. private MissionEndPoint MissionEndPoint;
  16. private CurrentLocation currentLocation;
  17. private double temptime = 0;
  18. private double probability = 0;
  19. private double finalProbability = 1.0;
  20. public bool isseePerson = false;
  21. private int fireIndex = -1; // 记录发现火点的位置
  22. public EquationHelper helper;
  23. public SeaSouJiuTask taskContent;
  24. public bool Success = false; //本目标搜救是否成功
  25. public GetNCData getNCData;
  26. public double resulttime;
  27. public SearchMissionPayload searchMissionPayload;
  28. //private bool isbool;
  29. //private bool isbool2;
  30. //Text_readNC text_ReadNC;
  31. public SearchMissionMode SearchMode;
  32. int Days;
  33. int Hour;
  34. public override void Reset()
  35. {
  36. base.Reset();
  37. IsOver = false;
  38. isseePerson = false;
  39. Success = false;
  40. SearchTime = 0;
  41. TotalTime = 0;
  42. resulttime = 0;
  43. }
  44. public override void Start()
  45. {
  46. int Year = Convert.ToInt32(taskContent.missionInformation.StartDate.Split("年")[0]);
  47. int Month = Convert.ToInt32(taskContent.missionInformation.StartDate.Split("年")[1].Split("月")[0]);
  48. int Day = Convert.ToInt32(taskContent.missionInformation.StartDate.Split("年")[1].Split("月")[1].Split("日")[0]);
  49. Hour = Convert.ToInt32(taskContent.missionInformation.StartTime.Split("时")[0]);
  50. Days = GetDaysInYear(Year, Month, Day);
  51. //Console.WriteLine("Day:" + Days);
  52. Velocitys = new double[5] { 220, 220, 60, 110, 0 }; // 速度
  53. //TODO 计算 AirRoute[]
  54. double[] initialPosition =
  55. {
  56. FlightPlanEditor.targetpoint[0].TargetPointLatitude, FlightPlanEditor.targetpoint[0].TargetPointLongitude
  57. };
  58. double dt = 1;
  59. double totalTime = 24.0;
  60. //if (!isbool2)
  61. //{
  62. Text_readNC text_ReadNC = new Text_readNC();
  63. text_ReadNC.initlatitudes = FlightPlanEditor.targetpoint[0].TargetPointLatitude;
  64. text_ReadNC.initlongitudes = FlightPlanEditor.targetpoint[0].TargetPointLongitude;
  65. //text_ReadNC = new Text_readNC();
  66. text_ReadNC.GetNCData();
  67. text_ReadNC.GetWaveHighData();
  68. //isbool2 = true;
  69. //}
  70. var nCread = text_ReadNC.windNCread;
  71. //漂移轨迹
  72. List<double[]> trajectory = SeaSJ.CalculateDriftTrajectory(nCread, initialPosition, dt, totalTime, text_ReadNC.times, text_ReadNC.latitudes, text_ReadNC.longitudes, text_ReadNC.times1, text_ReadNC.latitudes1, text_ReadNC.longitudes1, text_ReadNC.times2, text_ReadNC.latitudes2, text_ReadNC.longitudes2, text_ReadNC.times3, text_ReadNC.latitudes3, text_ReadNC.longitudes3, Days, Hour, text_ReadNC.initlatitudes, text_ReadNC.initlongitudes);
  73. // 生成任务终点
  74. MissionEndPoint = new MissionEndPoint
  75. {
  76. MissionEndPointLongitude = trajectory[^1][1],
  77. MissionEndPointLatitude = trajectory[^1][0],
  78. MissionEndPointHeight = 0
  79. };
  80. var temp = SeaSJ.getminEnclosingRect(trajectory);
  81. // temp 转成 List<Point>
  82. List<Point> points = new List<Point>();
  83. foreach (var item in temp)
  84. {
  85. points.Add(new Point(item[0], item[1]));
  86. }
  87. Point basePoint = new Point(FlightPlanEditor.originbase.BaseLatitude,
  88. FlightPlanEditor.originbase.BaseLongitude);
  89. List<Point> waypoints = new List<Point>();
  90. //*******
  91. if (SearchMode.SearchMode == "平行线搜索")
  92. {
  93. waypoints = ParallellineSearch.parallellineSearch(basePoint, points, SearchMode.SearchWidth);
  94. }
  95. if (SearchMode.SearchMode == "扇形搜索")
  96. {
  97. // 求 points 的中心点
  98. var startPoint1 = new Point
  99. {
  100. lat = points[0].lat / 2 + points[1].lat / 2,
  101. lon = points[0].lon / 2 + points[1].lon / 2
  102. };
  103. var startPoint2 = new Point
  104. {
  105. lat = points[2].lat / 2 + points[3].lat / 2,
  106. lon = points[2].lon / 2 + points[3].lon / 2
  107. };
  108. var centerPoint = new Point
  109. {
  110. lat = startPoint1.lat / 2 + startPoint2.lat / 2,
  111. lon = startPoint1.lon / 2 + startPoint2.lon / 2
  112. };
  113. waypoints = SectorSearch.sectorSearch(centerPoint, 30, SearchMode.SearchWidth);
  114. }
  115. if (SearchMode.SearchMode == "扩展矩形搜索")
  116. {
  117. // 求 points 的中心点
  118. var startPoint1 = new Point
  119. {
  120. lat = points[0].lat / 2 + points[1].lat / 2,
  121. lon = points[0].lon / 2 + points[1].lon / 2
  122. };
  123. var startPoint2 = new Point
  124. {
  125. lat = points[2].lat / 2 + points[3].lat / 2,
  126. lon = points[2].lon / 2 + points[3].lon / 2
  127. };
  128. var centerPoint = new Point
  129. {
  130. lat = startPoint1.lat / 2 + startPoint2.lat / 2,
  131. lon = startPoint1.lon / 2 + startPoint2.lon / 2
  132. };
  133. // 以 centerPoint 为中心,生成正方形区域
  134. var lonD = Math.Abs(points[0].lon - points[1].lon);
  135. var latD1 = Math.Abs(points[1].lat - points[2].lat);
  136. var max = Math.Max(lonD, latD1);
  137. // 以 distance 为边长 以 centerPoint 为中心点的正方形
  138. var temp0 = new Point(centerPoint.lat + max / 2, centerPoint.lon - max / 2);
  139. var temp1 = new Point(centerPoint.lat + max / 2, centerPoint.lon + max / 2);
  140. var temp2 = new Point(centerPoint.lat - max / 2, centerPoint.lon + max / 2);
  141. var temp3 = new Point(centerPoint.lat - max / 2, centerPoint.lon - max / 2);
  142. waypoints = TZFX.GenerateWaypoints(temp0, temp1, temp2, temp3, max, SearchMode.SearchWidth);
  143. }
  144. // List<Point> 转成 List<AirRoute>
  145. List<AirRoute> airRoutes = new List<AirRoute>();
  146. foreach (var item in waypoints)
  147. {
  148. airRoutes.Add(new AirRoute
  149. {
  150. AirRouteLatitude = item.lat,
  151. AirRouteLongitude = item.lon
  152. });
  153. }
  154. FlightPlanEditor.airroute = airRoutes.ToArray();
  155. MissionPoint missionPoint = new MissionPoint();
  156. missionPoint.MissionPointLatitude = waypoints[0].lat;
  157. missionPoint.MissionPointLongitude = waypoints[0].lon;
  158. FlightPlanEditor.missionpoint = missionPoint;
  159. FXJHGenerate.FromStartToMission(FlightPlanEditor, ref TurningPoints); //生成从起点到任务段起点的航路点
  160. FXJHGenerate.SeaSouJiu(FlightPlanEditor, ref TurningPoints);
  161. FXJHGenerate.FXJHTPDiedai(FlightPlanEditor, ref TurningPoints, Velocitys, FuelConsumptions);
  162. double time = 0; // 第一次 搜寻结束
  163. for (int i = 0; i < TurningPoints.Count - 2; i++)
  164. {
  165. time += TurningPoints[i].SegmentFlightTime;
  166. }
  167. double3 targetPoint = new double3(FlightPlanEditor.targetpoint[0].TargetPointLongitude,
  168. FlightPlanEditor.targetpoint[0].TargetPointLatitude,
  169. FlightPlanEditor.targetpoint[0].TargetPointHeight);
  170. string cityName = helper.getCityName(targetPoint.x, targetPoint.y);
  171. //var vis = helper.getVisibility(cityName, DateTime.Now.ToString("yyyy-MM-dd HH"));
  172. string year = taskContent.missionInformation.StartDate.Split("年")[0];
  173. string month = taskContent.missionInformation.StartDate.Split("年")[1].Split("月")[0];
  174. if (Convert.ToInt32(month) < 10) month = "0" + month;
  175. string day = taskContent.missionInformation.StartDate.Split("年")[1].Split("月")[1].Split("日")[0];
  176. if (Convert.ToInt32(day) < 10) day = "0" + day;
  177. //string hour = taskContent.missionInformation.StartTime.Split("时")[0];
  178. string date = year + "-" + month + "-" + day;// + " " + hour;
  179. //Console.Write(date);
  180. //Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH"));
  181. var vis = helper.getVisibilityByDb(targetPoint.x, targetPoint.y, date); //DateTime.Now.ToString("yyyy-MM-dd HH")
  182. //if (!isbool)
  183. //{
  184. getNCData = new GetNCData();
  185. getNCData.initlatitudes = FlightPlanEditor.targetpoint[0].TargetPointLatitude;
  186. getNCData.initlongitudes = FlightPlanEditor.targetpoint[0].TargetPointLongitude;
  187. getNCData.GetData();
  188. //isbool = true;
  189. //}
  190. Task.Run(() =>
  191. {
  192. bool isseePerson = false; // 输出表格平均救助时间没看到人是0看到人调用下面get_result_time_rope方法 //人员存活率需要重新算读取Editor搜救目标点初始坐标人数 //任务准备时间取值所有任务最小
  193. double temptime = 0; // 自增时间,每次增加1s
  194. CurrentLocation currentLocation = new CurrentLocation();
  195. double probability = 0;
  196. double finalProbability = 1.0;
  197. Random random = new Random();
  198. int fireIndex = -1; // 记录发现火点的位置
  199. double windSpeed = 0;
  200. do
  201. {
  202. if (!isseePerson && temptime >= time)
  203. {
  204. FXJHGenerate.SeaSouJiu(FlightPlanEditor, ref TurningPoints);
  205. FXJHGenerate.FXJHTPDiedai(FlightPlanEditor, ref TurningPoints, Velocitys, FuelConsumptions);
  206. time = 0;
  207. for (int i = 0; i < TurningPoints.Count - 2; i++)
  208. {
  209. time += TurningPoints[i].SegmentFlightTime;
  210. }
  211. Log.Info($"+++++++++++搜寻结束 TurningPoints Count: {TurningPoints.Count} 下次结束的时间 {time}++++++++++++++++");
  212. }
  213. (currentLocation, _) =
  214. FXJHGenerate.GetAllCurrentLocation(TurningPoints, temptime); // 获取飞机当前位置
  215. double3 aricraftPoint = new double3(currentLocation.CurrentLon, currentLocation.CurrentLat,
  216. currentLocation.CurrentHei);
  217. double3 targetPoint = new double3(FlightPlanEditor.targetpoint[0].TargetPointLongitude,
  218. FlightPlanEditor.targetpoint[0].TargetPointLatitude,
  219. FlightPlanEditor.targetpoint[0].TargetPointHeight);
  220. var wind = SeaSJ.GetWindVelocityFromAPI(nCread, currentLocation.CurrentLat, currentLocation.CurrentLon,
  221. temptime, text_ReadNC.times, text_ReadNC.latitudes, text_ReadNC.longitudes, text_ReadNC.times1, text_ReadNC.latitudes1, text_ReadNC.longitudes1, Days, Hour);
  222. windSpeed = Math.Sqrt(wind[0] * wind[0] + wind[1] * wind[1]);
  223. var waveHigh = SeaSJ.GetWaveHeightFromAPI(nCread, currentLocation.CurrentLon,
  224. currentLocation.CurrentLat, temptime, text_ReadNC.times4, text_ReadNC.latitudes4, text_ReadNC.longitudes4, Days, Hour);
  225. var distance = Utils.Util.GetDistance(currentLocation.CurrentLon, targetPoint.x,
  226. currentLocation.CurrentLat,
  227. targetPoint.y);
  228. Log.Info("距离:====================" + distance);
  229. if (searchMissionPayload.SearchPayload == "雷达搜索")
  230. {
  231. // Pd0 = 0.5 / Pf0 = Math.Pow(10,-6) / Pf = Math.Pow(10,-6) / R0 = 23645 / sigma0 = 5000 / sigma = Editor雷达截面面积 // R 单位m
  232. probability = helper.GetRadarPossibility(0.5, Math.Pow(10, -6), Math.Pow(10, -6), 23645, aricraftPoint, FlightPlanEditor.targetpoint[0], FlightPlanEditor.targetpoint[0].TargetType.RadarArea, 5000, vis);
  233. }
  234. else if (searchMissionPayload.SearchPayload == "目视搜素")
  235. {
  236. probability = helper.GetMushiSeaProbability(aricraftPoint, vis, waveHigh, FlightPlanEditor.targetpoint[0]);
  237. }
  238. else if (searchMissionPayload.SearchPayload == "光电(红外)搜索")
  239. {
  240. // Lt = 探测目标亮度 Editor / At = 探测目标面积 Editor / τa = 1 / Lb = 3 / A0 = 1 / D0 = 0.075 / Dstar = 3 / τo = 0.8 / Ad = 0.0073728 / Δf = 0.125 / δ = 0.5 / Pf0 = Math.Pow(10, -6);
  241. probability = helper.GetInfraredDetectionProbability(FlightPlanEditor.targetpoint[0].TargetType.TargetBrightness, FlightPlanEditor.targetpoint[0].TargetType.TargetArea, 1, 3, 1, aricraftPoint, FlightPlanEditor.targetpoint[0], 0.075, 3, 0.8, 0073728, 0.125, 0.5, Math.Pow(10, -6));
  242. }
  243. // probability = helper.getProbability(aricraftPoint, targetPoint,pb, currentLocation.CurrentCourse,
  244. // windSpeed, waveHigh, "落水人员", "海上"); // 计算发现概率,需要其他模型输入 // 计算发现概率,需要其他模型输入
  245. //finalProbability *= (1 - probability);
  246. // 到搜寻航路点的最后一个点 (?)
  247. Log.Info(
  248. $"海上任务:{taskContent.missionInformation.MissionName} 机型: {AircraftId} 当前时间:{temptime},当前位置:{currentLocation.CurrentLon},{currentLocation.CurrentLat},{currentLocation.CurrentHei},概率:{probability},是否看到落水人员:{isseePerson}");
  249. double randomValue = random.NextInt64(9006, 10000); // 生成随机数比较概率 //0.05 第一个参数越高概率越低 / 8985 0.212 100 / 9000 0.16 100 / 8995 0.14 100 / 8997 0.12 100 / 8999 0.18 100 / 9006 0.11 100 / 9010 0.12 100 / 9030 0.03 100 / 9026 0.07 0.04 100 / 9028 0.03 100 / 9027 0.04 100 / 9025 0.05 100 /
  250. //9025 0.02 100 / 9015 0.02 100 / 9020 0.04 100 / 9022 0.05 100 / 9022 0.04 100 / 9022 0 100 /9018 0.01 100 / 9021 0.01 100 / 9023 0.03 100 / 9000 0.08 100 / 9006
  251. randomValue /= 10000;
  252. if (randomValue < probability) // 1 - finalProbability
  253. {
  254. isseePerson = true;
  255. this.isseePerson = true;
  256. fireIndex = currentLocation.Currentsegnum; // 记录当前航路点位置
  257. IsOver = true;
  258. }
  259. else
  260. {
  261. isseePerson = false;
  262. }
  263. if (temptime >= taskContent.missionInformation.TaskEndConditions.TaskTime)
  264. {
  265. IsOver = true;
  266. isseePerson = false;
  267. }
  268. temptime += 10;
  269. } while (!isseePerson && IsOver == false);
  270. if (isseePerson)
  271. {
  272. for (int i = 0; i < TurningPoints.Count - 1; i++) // 总飞行时间
  273. {
  274. TotalTime += TurningPoints[i].SegmentFlightTime; // 总时间
  275. }
  276. double time = TotalTime; //time——搜索时间,单位:秒;数据测试用
  277. double latitude = FlightPlanEditor.targetpoint[0].TargetPointLatitude; //落水人员纬度;数据测试用
  278. double longitude = FlightPlanEditor.targetpoint[0].TargetPointLongitude; //落水人员经度,数据测试用
  279. double survivalTime = SurvivalTimeModel.SurvivalTime(getNCData.tempreadNC, latitude, longitude, time, text_ReadNC.times, text_ReadNC.latitudes, text_ReadNC.longitudes, Days, Hour); //幸存时间
  280. // SHJParameter.person_number = Eidtor里读取 SHJParameter.windspeed Nc/Editor目标气象信息读取时间先取任务初始时间
  281. resulttime = get_result_time_rope(30, FlightPlanEditor.targetpoint[0].TargetType.Count, windSpeed, vis, 0.75, 1.38).time; //调运时间
  282. if (survivalTime * 3600 > time)
  283. {
  284. Success = true;
  285. }
  286. else
  287. {
  288. Success = false;
  289. }
  290. Log.Info("TotalTime:" + TotalTime);
  291. Log.Info("幸存时间:" + survivalTime);
  292. }
  293. //Console.WriteLine(
  294. // $"海上任务1:{taskContent.missionInformation.MissionName} 机型: {AircraftId} 当前时间:{temptime},当前位置:{currentLocation.CurrentLon},{currentLocation.CurrentLat},{currentLocation.CurrentHei},概率:{probability},最终概率:{1 - finalProbability},是否看到落水人员:{isseePerson}");
  295. Log.Info(
  296. $"海上任务:{taskContent.missionInformation.MissionName} 机型: {AircraftId} 当前时间:{temptime},当前位置:{currentLocation.CurrentLon},{currentLocation.CurrentLat},{currentLocation.CurrentHei},概率:{probability},是否看到落水人员:{isseePerson},人员是否幸存:{Success}");
  297. //finalProbability = 1 - finalProbability;
  298. if (fireIndex != -1)
  299. {
  300. // 删除目标点位置开始的所有后续航路点
  301. TurningPoints.RemoveRange(fireIndex + 1, TurningPoints.Count - fireIndex - 1);
  302. MissionPoint missionPoint = new MissionPoint
  303. {
  304. MissionPointLongitude = currentLocation.CurrentLon,
  305. MissionPointLatitude = currentLocation.CurrentLat,
  306. MissionPointHeight = currentLocation.CurrentHei
  307. };
  308. FXJHGenerate.SeaSouJiu2(FlightPlanEditor, missionPoint, ref TurningPoints);
  309. //发现的目标坐标,需要其他模型输入
  310. FXJHGenerate.FromMissionToEnd(FlightPlanEditor, MissionEndPoint, ref TurningPoints);
  311. }
  312. FXJHGenerate.FXJHTPDiedai(FlightPlanEditor, ref TurningPoints, Velocitys, FuelConsumptions);
  313. End();
  314. });
  315. }
  316. public static handling_result get_result_time_rope(double H, int person_number, double windspeed, double vis, double upspeed, double downspeed)
  317. {
  318. handling_result result = new handling_result();
  319. if (windspeed < 8 && vis > 3)
  320. {
  321. result.time = (person_number * H) / downspeed + (person_number * H) / upspeed;
  322. result.success = true;
  323. }
  324. else
  325. {
  326. result.success = false;
  327. }
  328. return result;
  329. }
  330. public static int GetDaysInYear(int year, int month, int day)
  331. {
  332. int[] daysInMonths = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
  333. if (IsLeapYear(year))
  334. {
  335. daysInMonths[1] = 29;
  336. }
  337. int days = day;
  338. for (int i = 0; i < month - 1; i++)
  339. {
  340. days += daysInMonths[i];
  341. }
  342. return days;
  343. }
  344. public static bool IsLeapYear(int year)
  345. {
  346. return (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0);
  347. }
  348. public override void End()
  349. {
  350. for (int i = 0; i < TurningPoints.Count - 2; i++)
  351. {
  352. EffMisTime += TurningPoints[i].SegmentFlightTime;
  353. }
  354. for (int i = 0; i < TurningPoints.Count - 3; i++)
  355. {
  356. SearchTime += TurningPoints[i].SegmentFlightTime; //搜索时间
  357. }
  358. for (int i = 0; i < TurningPoints.Count; i++)
  359. {
  360. TotalTime += TurningPoints[i].SegmentFlightTime; // 总时间
  361. }
  362. TotalFuelConsumption = TurningPoints[0].RemainingFuel -
  363. TurningPoints[TurningPoints.Count - 1].RemainingFuel;
  364. //GetNCData getNCData = new GetNCData();
  365. ////getNCData.GetData();
  366. }
  367. }
  368. [ObjectSystem]
  369. public class AircraftSJAwakeSystem : AwakeSystem<AircraftSJ, FlightPlanEditor>
  370. {
  371. public override void Awake(AircraftSJ self, FlightPlanEditor flightPlanEditor)
  372. {
  373. self.FlightPlanEditor = flightPlanEditor;
  374. self.helper = new EquationHelper(HttpInterface.baseUrl);
  375. self.Awake();
  376. }
  377. }