73 lines
2.0 KiB
C#
73 lines
2.0 KiB
C#
using System.Collections.Concurrent;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using TaskExtensions = MesETL.Shared.Helper.TaskExtensions;
|
|
|
|
namespace MesETL.App.Services;
|
|
|
|
/// <summary>
|
|
/// 数据队列
|
|
/// </summary>
|
|
public class DataRecordQueue : IDisposable
|
|
{
|
|
private readonly BlockingCollection<DataRecord> _queue;
|
|
|
|
private long _currentCharCount;
|
|
private readonly long _maxCharCount = 2_147_483_648; // 4GiB
|
|
|
|
public int Count => _queue.Count;
|
|
public bool IsCompleted => _queue.IsCompleted;
|
|
public bool IsAddingCompleted => _queue.IsAddingCompleted;
|
|
|
|
public long LongestFieldCharCount { get; private set; }
|
|
|
|
public event Action? OnRecordWrite;
|
|
public event Action? OnRecordRead;
|
|
|
|
public DataRecordQueue() : this(500_000, 2_147_483_648) // 默认容量最大500K
|
|
{
|
|
}
|
|
|
|
public DataRecordQueue(int boundedCapacity, long maxCharCount)
|
|
{
|
|
_queue = new BlockingCollection<DataRecord>(boundedCapacity);
|
|
_maxCharCount = maxCharCount;
|
|
}
|
|
|
|
public void CompleteAdding() => _queue.CompleteAdding();
|
|
|
|
public bool TryDequeue([MaybeNullWhen(false)] out DataRecord record)
|
|
{
|
|
if (_queue.TryTake(out record))
|
|
{
|
|
Interlocked.Add(ref _currentCharCount, -record.FieldCharCount);
|
|
OnRecordRead?.Invoke();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
public async Task EnqueueAsync(DataRecord record)
|
|
{
|
|
var charCount = record.FieldCharCount;
|
|
LongestFieldCharCount = Math.Max(LongestFieldCharCount, charCount);
|
|
if (_currentCharCount + charCount > _maxCharCount)
|
|
{
|
|
// 不用Task.WaitUntil是为了防止产生Lambda闭包
|
|
while (!(_currentCharCount + charCount < _maxCharCount))
|
|
{
|
|
await Task.Delay(50);
|
|
}
|
|
}
|
|
_queue.Add(record);
|
|
Interlocked.Add(ref _currentCharCount, charCount);
|
|
OnRecordWrite?.Invoke();
|
|
}
|
|
|
|
|
|
public void Dispose()
|
|
{
|
|
_queue.Dispose();
|
|
}
|
|
} |