Commit 7fad816d authored by 潘栩锋's avatar 潘栩锋 🚴

添加 AD盒2021.B2 导出grid数据带有每个grid的时间

parent b29fe1ad
...@@ -13,6 +13,7 @@ namespace FlyADBase ...@@ -13,6 +13,7 @@ namespace FlyADBase
public int grid_start; public int grid_start;
public int[] buf; public int[] buf;
public int[] buf2; public int[] buf2;
public DateTime[] times;
} }
public delegate void MiniGridEventHandler(object sender, MiniGridEventArgs e); public delegate void MiniGridEventHandler(object sender, MiniGridEventArgs e);
} }
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
using Misc; using Misc;
using NLog.Fluent; using NLog.Fluent;
using System; using System;
using System.Collections.Generic;
using System.Diagnostics; using System.Diagnostics;
using System.Linq; using System.Linq;
using System.Reflection; using System.Reflection;
...@@ -73,17 +74,69 @@ namespace FlyADBase ...@@ -73,17 +74,69 @@ namespace FlyADBase
/// </summary> /// </summary>
public bool IsTimeToPushTimeGridAdv { get; private set; } = false; public bool IsTimeToPushTimeGridAdv { get; private set; } = false;
/// <summary>
/// 正向 Grid数组 更新时间
/// </summary>
public DateTime GridFrameUpdateTime_Forw { get; private set; }
/// <summary>
/// 反向 Grid数组 更新时间
/// </summary>
public DateTime GridFrameUpdateTime_Backw { get; private set; }
/// <summary> /// <summary>
/// 以timegrid 为单位,推送数据 /// 以timegrid 为单位,推送数据
/// </summary> /// </summary>
public event TimeGridAdv2EventHandler TimeGridAdv2Event; public event TimeGridAdv2EventHandler TimeGridAdv2Event;
#endregion #endregion
SGrid sGrid = new SGrid();
/// <summary>
/// 上一次更新AD,及推送timegrid的时间
/// </summary>
DateTime lastUpdateADTime = DateTime.MinValue;
/// <summary>
/// 上一次检查 更新grid数据 的时刻
/// </summary>
DateTime lastCheckUpdateGridTime = DateTime.MinValue;
/// <summary>
/// 上一次更新grid数组的最后时刻
/// </summary>
DateTime lastUpdateGridEndTime = DateTime.MinValue;
/// <summary>
/// 上一次更新grid数组的grid序号
/// </summary>
int lastUpdateGrid = -1;
/// <summary>
/// 上一次更新grid数组的方向
/// </summary>
DIRECTION lastUpdateGridDirection = DIRECTION.FIX;
/// <summary>
/// 上一次更新grid的序号
/// </summary>
int? lastPushMiniGrid = null;
/// <summary>
/// 上一次推送minigrid的方向
/// </summary>
DIRECTION lastPushMiniGridDirection = DIRECTION.FIX;
/// <summary>
/// 推送grid 的计时器,限制最小推送周期
/// </summary>
Stopwatch stopwatch_pushMiniGridTime = new Stopwatch();
/// <summary>
/// 记录下推送的时间
/// </summary>
DateTime lastPushTimeGridAdvTime = DateTime.MinValue;
DateTime lastUpdateADTime;
int lastGrid = Misc.MyBase.NULL_VALUE;
DateTime lastGridTime;
Stopwatch stopwatch_gridTime = new Stopwatch();
void advConstructor() void advConstructor()
{ {
...@@ -91,6 +144,8 @@ namespace FlyADBase ...@@ -91,6 +144,8 @@ namespace FlyADBase
this.PropertyChanged += FlyAD7_PropertyChanged1; this.PropertyChanged += FlyAD7_PropertyChanged1;
sGrid.SetSize(GridLen);
//轮询事件触发 //轮询事件触发
PollModule.Current.Poll_Config(PollModule.POLL_CONFIG.ADD, OnPoll); PollModule.Current.Poll_Config(PollModule.POLL_CONFIG.ADD, OnPoll);
} }
...@@ -110,14 +165,26 @@ namespace FlyADBase ...@@ -110,14 +165,26 @@ namespace FlyADBase
break; break;
} }
} }
else if (e.PropertyName == nameof(GridLen))
{
sGrid.SetSize(GridLen);
}
} }
void advAfterContected() void advAfterContected()
{ {
mTimeGridAdvHelper.Clear(); mTimeGridAdvHelper.Clear();
lastUpdateADTime = DateTime.MinValue; lastUpdateADTime = DateTime.MinValue;
lastGrid = Misc.MyBase.NULL_VALUE; lastCheckUpdateGridTime = DateTime.MinValue;
lastGridTime = DateTime.MinValue; lastUpdateGridEndTime = DateTime.MinValue;
lastUpdateGrid = -1;
lastUpdateGridDirection = DIRECTION.FIX;
lastPushMiniGrid = null;
lastPushMiniGridDirection = DIRECTION.FIX;
stopwatch_pushMiniGridTime.Reset();
lastPushTimeGridAdvTime = DateTime.MinValue;
} }
#region IFlyADClientAdv function #region IFlyADClientAdv function
...@@ -221,10 +288,18 @@ namespace FlyADBase ...@@ -221,10 +288,18 @@ namespace FlyADBase
/// <returns></returns> /// <returns></returns>
public TimeGridAdv2EventArgs GetTimeGridAdv2Event(DateTime beginTime) public TimeGridAdv2EventArgs GetTimeGridAdv2Event(DateTime beginTime)
{ {
if (mTimeGridAdvHelper.DataPool.Count() == 0)
return null;
//获取ad列表 //获取ad列表
var dataList = mTimeGridAdvHelper.GetData(beginTime); var dataList = mTimeGridAdvHelper.GetData(beginTime);
if (dataList.Count() == 0) if (dataList.Count() == 0)
return null; return null;
return GetTimeGridAdv2Event(dataList);
}
TimeGridAdv2EventArgs GetTimeGridAdv2Event(List<DateTimeUnit5> dataList)
{
DateTime endTime = mTimeGridAdvHelper.NewestTime; DateTime endTime = mTimeGridAdvHelper.NewestTime;
Misc.DIRECTION direction; Misc.DIRECTION direction;
...@@ -240,11 +315,17 @@ namespace FlyADBase ...@@ -240,11 +315,17 @@ namespace FlyADBase
} }
//数据滞后 //数据滞后
if(!TimeGridAdvHelperExt.AdLagCorr(dataList, ref endTime, ADLag, AD2Lag)) int lag = Math.Max(Math.Abs(ADLag), Math.Abs(AD2Lag));
if (dataList.Count() < lag * 5)
{ {
throw new Exception("数据量太少,无法滞后修正"); //数据量太少,无法滞后修正
}
}
else
{
TimeGridAdvHelperExt.AdLagCorr(dataList, ref endTime, ADLag, AD2Lag);
}
TimeGridAdv2EventArgs eventArgs = new TimeGridAdv2EventArgs(); TimeGridAdv2EventArgs eventArgs = new TimeGridAdv2EventArgs();
...@@ -253,7 +334,26 @@ namespace FlyADBase ...@@ -253,7 +334,26 @@ namespace FlyADBase
eventArgs.DataList = dataList; eventArgs.DataList = dataList;
return eventArgs; return eventArgs;
} }
public TimeGridAdv2EventArgs GetTimeGridAdv2Event(int dataCnt)
{
if (mTimeGridAdvHelper.DataPool.Count() == 0)
return null;
int adLag = Math.Max(Math.Abs(ADLag), Math.Abs(AD2Lag));
//获取ad列表
List<DateTimeUnit5> dataList;
int skipCnt = mTimeGridAdvHelper.DataPool.Count() - dataCnt - adLag;
if (skipCnt < 0)
{
dataList = mTimeGridAdvHelper.DataPool.ToList();
}
else
{
dataList = mTimeGridAdvHelper.DataPool.Skip(skipCnt).ToList();
}
return GetTimeGridAdv2Event(dataList);
}
void OnPoll() void OnPoll()
{ {
if (!IsConnected) if (!IsConnected)
...@@ -261,6 +361,7 @@ namespace FlyADBase ...@@ -261,6 +361,7 @@ namespace FlyADBase
if (!IsReady) if (!IsReady)
return; return;
OnPoll_grid();
OnPoll_TimeGrid(); OnPoll_TimeGrid();
OnPoll_TimeGridAdv(); OnPoll_TimeGridAdv();
OnPoll_miniGrid(); OnPoll_miniGrid();
...@@ -304,110 +405,499 @@ namespace FlyADBase ...@@ -304,110 +405,499 @@ namespace FlyADBase
lastUpdateADTime = Now; lastUpdateADTime = Now;
} }
//没有滞后处理,没有机架修正 int GetDataPoolGrid(int index)
void OnPoll_miniGrid() {
return (int)mTimeGridAdvHelper.DataPool[index].pos / PosOfGrid;
}
/// <summary>
/// 找到上一个grid的最后时刻
/// </summary>
/// <param name="lastValidIndex"></param>
/// <param name="updateGridBeginIndex"></param>
/// <param name="updateGridEndIndex"></param>
/// <returns></returns>
bool FindOutUpdateGridEndIndex(int lastValidIndex, int updateGridBeginIndex, out int updateGridEndIndex)
{ {
int grid = Position / PosOfGrid; updateGridEndIndex = updateGridBeginIndex;
//最后 有效数据 grid号
int lastValidGrid = GetDataPoolGrid(lastValidIndex);
if (!Misc.MyBase.ISVALIDATA(lastGrid)) int firstGrid = GetDataPoolGrid(updateGridBeginIndex);
if (firstGrid == lastValidGrid)
{ {
lastGrid = grid; //整个队列 都是同一个grid, 不处理
lastGridTime = Now; return false;
stopwatch_gridTime.Restart();
return;
} }
else
{
updateGridEndIndex = updateGridBeginIndex;
if (lastGrid == grid) //从 有效数据 向前找到 grid号不一样的
//肯定能找到, 最起码 [0] 是不一样的
for (int i = lastValidIndex; i >= 0; i--)
{ {
if (Now - lastGridTime > TimeSpan.FromSeconds(2)) int grid = GetDataPoolGrid(i);
if (grid != lastValidGrid)
{
//找到了
updateGridEndIndex = i;
break;
}
}
}
return true;
}
/// <summary>
/// 停下来了,且数据已经可以获取了
/// </summary>
/// <returns></returns>
public bool IsTimeToGetGridAfterStop()
{ {
//太慢,超过2秒还没走完一个grid if (mTimeGridAdvHelper.DriveStatusPool.Count() == 0)
//强制缩少数据 return false;//没有状态
lastGridTime = Now; var cell = mTimeGridAdvHelper.DriveStatusPool.Last();
if (cell.driveStatus == DRIVE_MAN_STATUS.RUNNING)
return false;//运行中
if (lastCheckUpdateGridTime >= cell.dt)
return true;
return false;
} }
void OnPoll_grid()
{
bool isStopTime = false;
//最后 时间点
DateTime newestTime = mTimeGridAdvHelper.NewestTime;
if (newestTime == DateTime.MinValue)//还没有 有效数据
return; return;
int lastIndex = mTimeGridAdvHelper.DataPool.Count() - 1;
//当前 更新grid数组 开始的数据池序号
int updateGridBeginIndex;
//当前 更新grid数组 结束的数据池序号
int updateGridEndIndex = -1;
//上一次更新grid数组的最后时刻
if (lastUpdateGridEndTime == DateTime.MinValue)
{
//从来没有处理完数据
updateGridBeginIndex = 0;
}
else
{
if (lastUpdateGridEndTime >= newestTime)
{
return;//不需要更新
} }
if (stopwatch_gridTime.Elapsed < TimeSpan.FromSeconds(0.1))
int lastUpdateGridEndIndex = mTimeGridAdvHelper.Time2Index(lastUpdateGridEndTime);
updateGridBeginIndex = lastUpdateGridEndIndex + 1;
if (updateGridBeginIndex < 0)
{ {
//太快,控制推送频率 //很久以前了,复位
updateGridBeginIndex = 0;
}
if (mTimeGridAdvHelper.DriveStatusPool.Count() != 0)
{
var cell = mTimeGridAdvHelper.DriveStatusPool.Last();
if (cell.driveStatus != DRIVE_MAN_STATUS.RUNNING)
{
//当前已经停下来
DateTime stopTime = cell.dt;
if (stopTime > lastUpdateGridEndTime)
{
//把最后一个grid也导出
if (newestTime >= stopTime)
{
//当前的有效时间已经大于停止时间
updateGridEndIndex = lastIndex;
isStopTime = true;
}
}
}
}
}
//记录时间,已经调用过了
lastCheckUpdateGridTime = newestTime;
if (updateGridEndIndex == -1)
{
//找到上一个grid的最后时刻
bool ret = FindOutUpdateGridEndIndex(lastIndex, updateGridBeginIndex, out updateGridEndIndex);
if (!ret)//没有不一样的grid序号
return; return;
} }
stopwatch_gridTime.Restart();
int firstGrid = GetDataPoolGrid(updateGridBeginIndex);
//最后 有效数据 grid号
int lastValidGrid = GetDataPoolGrid(lastIndex);
//方向 //方向
Misc.DIRECTION direction; Misc.DIRECTION direction;
if (lastGrid < grid) if (firstGrid < lastValidGrid)
direction = Misc.DIRECTION.FORWARD; direction = Misc.DIRECTION.FORWARD;
else else
direction = Misc.DIRECTION.BACKWARD; direction = Misc.DIRECTION.BACKWARD;
int dir_index = direction == Misc.DIRECTION.FORWARD ? 0 : 1;
if (lastUpdateGridEndTime != DateTime.MinValue)
{
if (lastUpdateGridDirection != direction)
{
//方向变化了,清除当前方向数据
sGrid.Clear(dir_index);
}
}
//grid号列表
List<int> gridList = new List<int>();
//grid ad列表
List<int>[] adLists = new List<int>[2] { new List<int>(), new List<int>() };
DateTime endTime = DateTime.MinValue;
int lastGrid = -1;
int grid_begin = GetDataPoolGrid(updateGridBeginIndex);
int grid_end = GetDataPoolGrid(updateGridEndIndex);
logger.Debug($"OnPoll_grid() dir_index={dir_index} grid=[{grid_begin}~{grid_end}]");
for (int i = updateGridBeginIndex; i <= updateGridEndIndex; i++)
{
int grid = GetDataPoolGrid(i);
if (i != updateGridBeginIndex)
{
if (grid != lastGrid)
{
if (lastGrid >= 0 && lastGrid < GridLen)
{
sGrid.GetData(0)[dir_index][lastGrid] = (int)adLists[0].Average();
sGrid.GetData(1)[dir_index][lastGrid] = (int)adLists[1].Average();
sGrid.times[dir_index][lastGrid] = endTime;
gridList.Add(lastGrid);
}
adLists[0].Clear();
adLists[1].Clear();
}
}
var ads = new int?[] {
mTimeGridAdvHelper.DataPool[i].ad,
mTimeGridAdvHelper.DataPool[i].ad2
};
if (ads.Any(a => a == null))
{
//居然没有数据
throw new Exception($"DataPool[{i}].ad0 或 ad1 没有AD数据, grid = {grid}");
}
for (int j = 0; j < 2; j++)
{
var dataList = mTimeGridAdvHelper.GetData(lastGridTime); adLists[j].Add((int)ads[j]);
var endTime = mTimeGridAdvHelper.NewestTime; //1个grid的数据不能大于2000ms
if (adLists[j].Count() > 2000)
adLists[j].RemoveAt(0);
}
endTime = mTimeGridAdvHelper.Index2Time(i);
lastGrid = grid;
}
//数据滞后
if (!TimeGridAdvHelperExt.AdLagCorr(dataList, ref endTime, ADLag, AD2Lag))
{ {
//throw new Exception("数据量太少,无法滞后修正"); //最后一个lastGrid 肯定是满数据的。 下一个时间, grid序号就会改变, 这个是在 FindOutUpdateGridEndIndex 操作的
return; if (lastGrid >= 0 && lastGrid < GridLen)
{
sGrid.GetData(0)[dir_index][lastGrid] = (int)adLists[0].Average();
sGrid.GetData(1)[dir_index][lastGrid] = (int)adLists[1].Average();
sGrid.times[dir_index][lastGrid] = endTime;
gridList.Add(lastGrid);
} }
adLists[0].Clear();
adLists[1].Clear();
}
//上一次 grid 与 当前grid 中间有一段空白
//线性插值 中间空白区
if (lastUpdateGridEndTime != DateTime.MinValue)
{
if (lastUpdateGridDirection == direction)
{
int grid0 = lastUpdateGrid;
int grid1 = firstGrid;
if (grid0 >= 0 && grid0 < GridLen &&
grid1 >= 0 && grid1 < GridLen)
{
if (Math.Abs(grid0 - grid1) > 1)//有空白区
{
FillEmptyGrid(dir_index, grid0, grid1);
logger.Debug($"OnPoll_grid() dir_index={dir_index} FillEmptyGrid({dir_index},{grid0},{grid1})");
}
}
}
}
//这次grid填充, 检查是否连续,如果有留空, 插值填充
for (int i = 1; i < gridList.Count(); i++)
{
int grid1 = gridList[i];
int grid0 = gridList[i - 1];
if (Math.Abs(grid1 - grid0) > 1) //有空白区
{
FillEmptyGrid(dir_index, grid0, grid1);
logger.Debug($"OnPoll_grid() dir_index={dir_index} FillEmptyGrid({dir_index},{grid0},{grid1})");
}
}
//获取grid图
TimeGridAdvHelperExt.ToGrid(dataList, PosOfGrid, GridLen,
out int grid_start, out int grid_end,
out int[] gridsOfAd1, out int[] gridsOfAd2,
isMiniGrid: true);
if (gridsOfAd1 == null)
lastUpdateGridEndTime = mTimeGridAdvHelper.Index2Time(updateGridEndIndex);
lastUpdateGridDirection = direction;
lastUpdateGrid = lastGrid;
if (isStopTime) {
if (direction == DIRECTION.FORWARD)
{
GridFrameUpdateTime_Forw = lastUpdateGridEndTime;
}
else {
GridFrameUpdateTime_Backw = lastUpdateGridEndTime;
}
}
}
/// <summary>
/// 填充 grid0 ~grid1 之间的空白
/// </summary>
/// <param name="dir_index"></param>
/// <param name="grid0"></param>
/// <param name="grid1"></param>
void FillEmptyGrid(int dir_index, int grid0, int grid1)
{
var adData0 = sGrid.GetData(0)[dir_index];//探头#0 dirIndex 方向 grid数据
var adData1 = sGrid.GetData(1)[dir_index];//探头#1 dirIndex 方向 grid数据
int ad0_0 = adData0[grid0];//探头#0 两个grid 数据 开始点
int ad0_1 = adData0[grid1];//探头#0 两个grid 数据 结束点
int ad1_0 = adData1[grid0];
int ad1_1 = adData1[grid1];
DateTime time0 = sGrid.times[dir_index][grid0];
DateTime time1 = sGrid.times[dir_index][grid1];
if (grid0 < grid1)
{
for (int i = grid0 + 1; i < grid1; i++)
{
int grid = i;
int ad = ad0_0 + (ad0_1 - ad0_0) * (grid - grid0) / (grid1 - grid0);
adData0[grid] = ad;
ad = ad1_0 + (ad1_1 - ad1_0) * (grid - grid0) / (grid1 - grid0);
adData1[grid] = ad;
DateTime time = time0 + TimeSpan.FromMilliseconds(((time1 - time0).Milliseconds * (grid - grid0) / (grid1 - grid0)));
sGrid.times[dir_index][grid] = time;
}
}
else
{
for (int i = grid1 + 1; i < grid0; i++)
{
int grid = i;
int ad = ad0_0 + (ad0_1 - ad0_0) * (grid - grid0) / (grid1 - grid0);
adData0[grid] = ad;
ad = ad1_0 + (ad1_1 - ad1_0) * (grid - grid0) / (grid1 - grid0);
adData1[grid] = ad;
DateTime time = time0 + TimeSpan.FromMilliseconds(((time1 - time0).Milliseconds * (grid - grid0) / (grid1 - grid0)));
sGrid.times[dir_index][grid] = time;
}
}
}
IEnumerable<T> Slice<T>(IEnumerable<T> list, int startIdx, int endIdx)
{
return list.Skip(startIdx).Take(endIdx - startIdx + 1);
}
void OnPoll_miniGrid()
{
if (MiniGridEvent == null)//没有注册事件
{
lastPushMiniGrid = null;//复位最后推送grid号
stopwatch_pushMiniGridTime.Stop();//停止计时
return; return;
}
if (lastUpdateGridEndTime == DateTime.MinValue)
return;//没有生成过一个grid
if (grid_start >= GridLen) if (lastPushMiniGrid == null)
{
//从来没有推送
}
else
{
if (lastPushMiniGrid == lastUpdateGrid)
{
return;//一样不用推送
}
if (stopwatch_pushMiniGridTime.IsRunning && stopwatch_pushMiniGridTime.Elapsed < TimeSpan.FromSeconds(0.1))
{
//太快,控制推送频率
return; return;
}
}
if (lastUpdateGrid >= 0 && lastUpdateGrid < GridLen)
{
//在范围内
//需要推送grid //需要推送grid
MiniGridEvent?.Invoke(this, new MiniGridEventArgs() if (lastPushMiniGrid == null || //从来没有推送
lastPushMiniGrid < 0 || lastPushMiniGrid >= GridLen ||//之前在范围外
lastPushMiniGridDirection != lastUpdateGridDirection)//之前的方向与现在不一样
{
PushMiniGridEvent(lastUpdateGridDirection, lastUpdateGrid);
}
else
{
//之前在范围内
if (lastUpdateGridDirection == DIRECTION.FORWARD)
{
if (lastPushMiniGrid < lastUpdateGrid)
{
PushMiniGridEvent(lastUpdateGridDirection, (int)lastPushMiniGrid + 1, lastUpdateGrid);
}
else
{
//逻辑不对, 之前的推送位置 在 最新位置 右边
PushMiniGridEvent(lastUpdateGridDirection, lastUpdateGrid);
}
}
else
{
if (lastPushMiniGrid > lastUpdateGrid)
{
PushMiniGridEvent(lastUpdateGridDirection, lastUpdateGrid, (int)lastPushMiniGrid - 1);
}
else
{
//逻辑不对, 之前的推送位置 在 最新位置 左边
PushMiniGridEvent(lastUpdateGridDirection, lastUpdateGrid);
}
}
}
}
lastPushMiniGridDirection = lastUpdateGridDirection;
lastPushMiniGrid = lastUpdateGrid;
stopwatch_pushMiniGridTime.Restart();
}
void PushMiniGridEvent(DIRECTION direction, int grid)
{
int dir_index = direction == Misc.DIRECTION.FORWARD ? 0 : 1;
var buf = new int[] { sGrid.data[dir_index][grid] };
var buf2 = new int[] { sGrid.data2[dir_index][grid] };
var times = new DateTime[] { sGrid.times[dir_index][grid] };
MiniGridEvent(this, new MiniGridEventArgs()
{ {
direction = direction, direction = direction,
grid_start = grid_start, grid_start = grid,
posOfGrid = PosOfGrid, posOfGrid = PosOfGrid,
buf = gridsOfAd1, buf = buf,
buf2= gridsOfAd2, buf2 = buf2,
times = times
}); });
lastGrid = grid;
lastGridTime = Now;
} }
void PushMiniGridEvent(DIRECTION direction, int beginGrid, int endGrid)
{
int dir_index = direction == Misc.DIRECTION.FORWARD ? 0 : 1;
void OnPoll_TimeGridAdv() var buf = Slice(sGrid.data[dir_index], beginGrid, endGrid);
var buf2 = Slice(sGrid.data2[dir_index], beginGrid, endGrid);
var times = Slice(sGrid.times[dir_index], beginGrid, endGrid);
MiniGridEvent(this, new MiniGridEventArgs()
{ {
if (!IsTimeToPushTimeGridAdv) direction = direction,
return; grid_start = beginGrid,
posOfGrid = PosOfGrid,
buf = buf.ToArray(),
buf2 = buf2.ToArray(),
times = times.ToArray()
IsTimeToPushTimeGridAdv = false; });
}
void OnPoll_TimeGridAdv()
{
if (TimeGridAdv2Event == null) if (TimeGridAdv2Event == null)
return; return;
//触发全部高级版的 timegrid if (mTimeGridAdvHelper.DriveStatusPool.Count() < 2)//起码要有 RUNNING, STOP 状态
bool ret = mTimeGridAdvHelper.GetLastRunningTime(out DateTime beginTime, out DateTime endTime, out DRIVE_MAN_ORDER order);
if (!ret)
return; return;
var cell = mTimeGridAdvHelper.DriveStatusPool.Last();
if (cell.driveStatus == DRIVE_MAN_STATUS.RUNNING)
return;//还在运行中
IsTimeToPushTimeGridAdv = false;
//当前已经停下来
DateTime stopTime = cell.dt;
if (lastPushTimeGridAdvTime !=DateTime.MinValue
&& lastPushTimeGridAdvTime>=stopTime)
{
return;//已经推送了出去
}
var cell2 = mTimeGridAdvHelper.DriveStatusPool[mTimeGridAdvHelper.DriveStatusPool.Count() - 2];
if (cell2.driveStatus != DRIVE_MAN_STATUS.RUNNING)
{
//异常, stop 之前不是running
throw new Exception($"异常, stop 之前不是running, cell2.driveStatus={cell2.driveStatus}");
}
//开始RUNNING时间
DateTime beginTime = cell2.dt;
var eventArgs = GetTimeGridAdv2Event(beginTime); var eventArgs = GetTimeGridAdv2Event(beginTime);
//记录下推送的时间,防止再次推送
lastPushTimeGridAdvTime = eventArgs.EndTime;
//获取ad列表 //获取ad列表
var dataList = eventArgs.DataList; var dataList = eventArgs.DataList;
int dataCnt = dataList.Count(); int dataCnt = dataList.Count();
if (dataCnt == 0) if (dataCnt == 0)
return; return;
endTime = eventArgs.EndTime; var endTime = eventArgs.EndTime;
var direction = eventArgs.Direction; var direction = eventArgs.Direction;
//生成grid图
TimeGridAdvHelperExt.ToGrid(dataList, PosOfGrid, GridLen,
out int grid_start, out int grid_end,
out int[] gridsOfAd1, out int[] gridsOfAd2);
//触发事件 //触发事件
TimeGridAdv2Event?.Invoke(this, new TimeGridAdv2EventArgs TimeGridAdv2Event?.Invoke(this, new TimeGridAdv2EventArgs
...@@ -417,6 +907,87 @@ namespace FlyADBase ...@@ -417,6 +907,87 @@ namespace FlyADBase
DataList= dataList DataList= dataList
}); });
} }
/// <summary>
/// 从正反缓存区, 获取grid数据
/// </summary>
/// <param name="direction">方向, 只有 正,反</param>
/// <param name="grid_start">grid 开始位置</param>
/// <param name="grid_len">grid 长度</param>
/// <param name="dat">grid 数据</param>
public void GetGrid(Misc.DIRECTION direction, out int[] buf0, out int[] buf1, out DateTime[] times)
{
int dirIndex = (direction == Misc.DIRECTION.FORWARD) ? 0 : 1;
buf0 = sGrid.GetData(0)[dirIndex].ToArray();
buf1 = sGrid.GetData(1)[dirIndex].ToArray();
times = sGrid.times[dirIndex].ToArray();
}
/// <summary>
/// 从正反缓存区, 获取grid数据
/// </summary>
/// <param name="direction">方向, 只有 正,反</param>
/// <param name="grid_start">grid 开始位置</param>
/// <param name="grid_len">grid 长度</param>
/// <param name="dat">grid 数据</param>
public void GetGrid(int adIndex, Misc.DIRECTION direction, int grid_start, int grid_len, out int[] dat, out DateTime[] times)
{
int dirIndex = (direction == Misc.DIRECTION.FORWARD) ? 0 : 1;
dat = new int[grid_len];
times = new DateTime[grid_len];
var adData = sGrid.GetData(adIndex);//获取序号为 adIndex 探头的正反向数据
var dirData = adData[dirIndex];//获取 该方向数据
var timeData = sGrid.times[dirIndex];
for (int i = 0; i < dat.Length; i++)
{
int grid_num = grid_start + i;
if (grid_num >= dirData.Length)
{
//剩余全部填入 NULL_VALUE;
for (int j = i; j < dat.Length; j++)
{
dat[i] = Misc.MyBase.NULL_VALUE;
times[i] = DateTime.MinValue;
}
break;
}
dat[i] = dirData[grid_num];
times[i] = timeData[grid_num];
}
return;
}
public void GetGrid(int adIndex, Misc.DIRECTION direction, int grid_start, int grid_len, out int[] dat)
{
GetGrid(adIndex, direction, grid_start, grid_len, out dat, out _);
}
/// <summary>
/// 从正反缓存区, 获取全部grid数据
/// </summary>
/// <param name="direction">方向, 只有 正,反</param>
/// <param name="dat">grid 数据</param>
public void GetGrid(int adIndex, Misc.DIRECTION direction, out int[] dat)
{
GetGrid(adIndex, direction, 0, GridLen, out dat);
}
/// <summary>
/// 从正反缓存区, 获取全部grid数据
/// </summary>
/// <param name="direction">方向, 只有 正,反</param>
/// <param name="dat">grid 数据</param>
public void GetGrid(int adIndex, Misc.DIRECTION direction, out int[] dat, out DateTime[] times)
{
GetGrid(adIndex, direction, 0, GridLen, out dat, out times);
}
} }
class CalSpeed class CalSpeed
...@@ -533,8 +1104,13 @@ namespace FlyADBase ...@@ -533,8 +1104,13 @@ namespace FlyADBase
class SGrid class SGrid
{ {
public const int GRID_MAX_SIZE = 1000; public const int GRID_MAX_SIZE = 1000;
int size; public int Size;
public int[][] data = new int[2][];//data[0]=forword, data[1]=backward
public int[][] data = new int[2][];//探头#1 data0[0]=forword, data0[1]=backward
public int[][] data2 = new int[2][];//探头#2 data1[0]=forword, data1[1]=backward
public DateTime[][] times = new DateTime[2][];//每个grid的时间 times[0]=forword, times[1]=backward
public SGrid() public SGrid()
{ {
//清空所有数据 //清空所有数据
...@@ -542,18 +1118,40 @@ namespace FlyADBase ...@@ -542,18 +1118,40 @@ namespace FlyADBase
} }
public void SetSize(int size) public void SetSize(int size)
{ {
this.size = size; this.Size = size;
data[0] = new int[size]; data[0] = new int[size];
data[1] = new int[size]; data[1] = new int[size];
data2[0] = new int[size];
data2[1] = new int[size];
times[0] = new DateTime[size];
times[1] = new DateTime[size];
Clear(); Clear();
} }
public int[][] GetData(int dataIndex)
{
if (dataIndex == 0)
return data;
else
return data2 ;
}
public void Clear() public void Clear()
{ {
//清空所有数据 //清空所有数据
for (int i = 0; i < size; i++) Clear(0);
Clear(1);
}
public void Clear(int index)
{
//清空所有数据
for (int i = 0; i < Size; i++)
{ {
data[0][i] = Misc.MyBase.NULL_VALUE; data[index][i] = Misc.MyBase.NULL_VALUE;
data[1][i] = Misc.MyBase.NULL_VALUE; data2[index][i] = Misc.MyBase.NULL_VALUE;
times[index][i] = DateTime.MinValue;
} }
} }
} }
......
...@@ -98,12 +98,39 @@ namespace FlyADBase ...@@ -98,12 +98,39 @@ namespace FlyADBase
/// </summary> /// </summary>
bool IsTimeToPushTimeGridAdv { get; } bool IsTimeToPushTimeGridAdv { get; }
/// <summary>
/// 停下来了,且数据已经可以获取了
/// </summary>
/// <returns></returns>
bool IsTimeToGetGridAfterStop();
/// <summary>
/// 正向 Grid数组 更新时间
/// </summary>
DateTime GridFrameUpdateTime_Forw { get; }
/// <summary>
/// 反向 Grid数组 更新时间
/// </summary>
DateTime GridFrameUpdateTime_Backw { get; }
/// <summary> /// <summary>
/// 以timegrid 为单位,推送数据 /// 以timegrid 为单位,推送数据
/// </summary> /// </summary>
event TimeGridAdv2EventHandler TimeGridAdv2Event; event TimeGridAdv2EventHandler TimeGridAdv2Event;
TimeGridAdv2EventArgs GetTimeGridAdv2Event(DateTime beginTime); TimeGridAdv2EventArgs GetTimeGridAdv2Event(DateTime beginTime);
TimeGridAdv2EventArgs GetTimeGridAdv2Event(int dataCnt);
/// <summary>
/// 从正反缓存区, 获取grid数据
/// </summary>
/// <param name="direction">方向, 只有 正,反</param>
/// <param name="grid_start">grid 开始位置</param>
/// <param name="grid_len">grid 长度</param>
/// <param name="dat">grid 数据</param>
void GetGrid(Misc.DIRECTION direction, out int[] buf0, out int[] buf1, out DateTime[] times);
void Save(); void Save();
bool Load(); bool Load();
......
...@@ -74,6 +74,18 @@ namespace FlyADBase ...@@ -74,6 +74,18 @@ namespace FlyADBase
{ {
Clear(); Clear();
} }
public int Time2Index(DateTime time)
{
if (time == DateTime.MinValue)
return -1;
return TimeGridAdvHelperExt.Time2Index(DataPool.Count(), NewestTime, time);
}
public DateTime Index2Time(int index)
{
return TimeGridAdvHelperExt.Index2Time(DataPool.Count(), NewestTime, index);
}
public class AddDataResponse public class AddDataResponse
{ {
public bool isChanged => isPosChanged || isPos2Changed || isIstatusChanged; public bool isChanged => isPosChanged || isPos2Changed || isIstatusChanged;
...@@ -103,9 +115,8 @@ namespace FlyADBase ...@@ -103,9 +115,8 @@ namespace FlyADBase
if (DataPool.Count()==0) if (DataPool.Count()==0)
{ {
//第1次添加 //第1次添加
NewestTime = dt;
DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus }); DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus });
NewestTime = dt;
//记录脉冲改变点 //记录脉冲改变点
positionChangedTime = NewestTime; positionChangedTime = NewestTime;
...@@ -116,11 +127,11 @@ namespace FlyADBase ...@@ -116,11 +127,11 @@ namespace FlyADBase
if (dt < NewestTime && NewestTime - dt > TimeSpan.FromMilliseconds(2)) //时间异常,居然早2秒了,数据全部删除 if (dt < NewestTime && NewestTime - dt > TimeSpan.FromMilliseconds(2)) //时间异常,居然早2秒了,数据全部删除
{ {
//TODO, 不能没有状态
Clear(); Clear();
NewestTime = dt;
DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus }); DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus });
NewestTime = dt;
//记录脉冲改变点 //记录脉冲改变点
positionChangedTime = NewestTime; positionChangedTime = NewestTime;
...@@ -131,11 +142,11 @@ namespace FlyADBase ...@@ -131,11 +142,11 @@ namespace FlyADBase
if ( dt > NewestTime && (dt - NewestTime > TimeSpan.FromSeconds(0.5)))//超过0.5s没放数据,异常 if ( dt > NewestTime && (dt - NewestTime > TimeSpan.FromSeconds(0.5)))//超过0.5s没放数据,异常
{ {
//TODO, 不能没有状态
Clear(); Clear();
NewestTime = dt;
DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus }); DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus });
NewestTime = dt;
//记录脉冲改变点 //记录脉冲改变点
positionChangedTime = NewestTime; positionChangedTime = NewestTime;
...@@ -150,7 +161,6 @@ namespace FlyADBase ...@@ -150,7 +161,6 @@ namespace FlyADBase
DataPool.RemoveAt(DataPool.Count() - 1); DataPool.RemoveAt(DataPool.Count() - 1);
DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus }); DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus });
NewestTime = dt; NewestTime = dt;
} }
else else
...@@ -169,6 +179,7 @@ namespace FlyADBase ...@@ -169,6 +179,7 @@ namespace FlyADBase
//填充完了,放入新的数据 //填充完了,放入新的数据
DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus }); DataPool.Add(new DateTimeUnit5() { ad = ad, ad2 = ad2, pos = postion, pos2 = postion2, istatus = istatus });
NewestTime = last_dt; NewestTime = last_dt;
break; break;
} }
} }
...@@ -535,6 +546,17 @@ namespace FlyADBase ...@@ -535,6 +546,17 @@ namespace FlyADBase
public static class TimeGridAdvHelperExt public static class TimeGridAdvHelperExt
{ {
public static int Time2Index(int dataCount, DateTime newestTime, DateTime time)
{
//TODO .TotalMilliseconds 是double,有机会出现0.99999问题
int index = dataCount - 1 - (int)(newestTime - time).TotalMilliseconds;
return index;
}
public static DateTime Index2Time(int dataCount, DateTime newestTime, int index)
{
return newestTime - TimeSpan.FromMilliseconds(dataCount - 1 - index);
}
/// <summary> /// <summary>
/// 导出的数据间隔是1ms 一个 /// 导出的数据间隔是1ms 一个
/// </summary> /// </summary>
...@@ -543,8 +565,10 @@ namespace FlyADBase ...@@ -543,8 +565,10 @@ namespace FlyADBase
/// <returns></returns> /// <returns></returns>
public static List<DateTimeUnit5> GetData(List<DateTimeUnit5> dataPool, DateTime newestTime, DateTime begin) public static List<DateTimeUnit5> GetData(List<DateTimeUnit5> dataPool, DateTime newestTime, DateTime begin)
{ {
int begin_idx = dataPool.Count() - (int)((newestTime - begin).TotalMilliseconds); //时间转为序号
int begin_idx = Time2Index(dataPool.Count(), newestTime, begin) + 1;
//限制序号位置
if (begin_idx < 0) if (begin_idx < 0)
begin_idx = 0; begin_idx = 0;
if (begin_idx > dataPool.Count() - 1) if (begin_idx > dataPool.Count() - 1)
...@@ -566,8 +590,10 @@ namespace FlyADBase ...@@ -566,8 +590,10 @@ namespace FlyADBase
/// <returns></returns> /// <returns></returns>
public static List<DateTimeUnit5> GetData(List<DateTimeUnit5> dataPool, DateTime newestTime, DateTime begin, DateTime end, out DateTime response_endTime) public static List<DateTimeUnit5> GetData(List<DateTimeUnit5> dataPool, DateTime newestTime, DateTime begin, DateTime end, out DateTime response_endTime)
{ {
int begin_idx = dataPool.Count() - (int)((newestTime - begin).TotalMilliseconds); //时间转为序号
int begin_idx = Time2Index(dataPool.Count(), newestTime, begin) + 1;
//限制序号位置
if (begin_idx < 0) if (begin_idx < 0)
begin_idx = 0; begin_idx = 0;
if (begin_idx > dataPool.Count() - 1) if (begin_idx > dataPool.Count() - 1)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment