这个方法最好是一个单独的不受别的控制器影响的class。因为你如果声明错位置的话一不小心就会把我们的队列重置。导致队列内容丢失造成损失。大家如果有某些方法运行时间太长但又需要快速给出结果,就可以利用队列,我们这边直接给她结果。然后具体操作我们在内部慢慢执行。
// 比如说我这里新建一个Class专门用来存放这个方法和队列内容容器
public class QueueTesting
{
// 全局变量 容器类型。其中ReqParameters是我生命的一个实体类,我要这个队列里面存储的是这个类型。当然大家需要什么类型就存储什么类型就行了。
public static ConcurrentQueue<ReqParameters> requestQueue = new ConcurrentQueue<ReqParameters>();
// 构造函数 可有可无,目前我这里并没有用构造函数一致初始化。后续中直接在Startup中注册一下服务
public QueueTesting
{
}
// 接下来就是具体方法了
public async Task ProcessRequestQueue(CancellationToken cancellationToken)
{
while (true)
{
// 检查队列是否有请求,并将请求从队列中取出
if (requestQueue.TryDequeue(out ReqRoomStatus_Price_timeandbhid request))
{
// 取出我们想要的参数,也就是上述我说的我生命的实体类中的属性
string id = request.id;
// 通过ID从 _odb 对象中异步获取订单信息
// 这也是某些方法我查了一下数据库读取出我想要的数据根据id
Object object = await _odb.OrdersByOidAsync(id);
// 检查订单是否为空
if (object == null)
{
// 如果我查找的对象为空,记录错误日志
Logger.LogError($"错误日志记录随意记录!!!!!");
}
else
{
//!!!!!!!!!!!!! 这一步就是我们要队列具体执行的方法了!!!!!!
// 就是因为我这个内部方法耗时严重!!所以我才要创建队列领出来单独执行的
// 如果对象不为空,使用 _orderUtil 对象异步提交对象
order = await _orderUtil.SubmitOrderAsync(order);
//!!!!!!!!!!!!! 这一步就是我们要队列具体执行的方法了!!!!!!
}
}
else
{
// 当队列为空时,等待一段时间再继续处理
await Task.Delay(1000, cancellationToken);
}
// 如果你希望在队列为空时跳出循环,可以取消注释以下代码
//if (requestQueue.IsEmpty)
//{
// break;
//}
}
}
}
后面我们要在Startup里面初始化注册一下这个服务。
public class RequestQueueBackgroundService : BackgroundService
{
private readonly QueueTesting _requestQueueController;
// 构造函数,接受一个 QueueTesting 对象作为参数
public RequestQueueBackgroundService(QueueTesting requestQueueController)
{
_requestQueueController = requestQueueController;
}
// 重写 BackgroundService 类中的 ExecuteAsync 方法
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
// 调用 _requestQueueController 对象的 ProcessRequestQueue 方法,传入 stoppingToken 参数
await _requestQueueController.ProcessRequestQueue(stoppingToken);
}
}
在Startup里面注册一个我们刚才重写的服务
services.AddHostedService<RequestQueueBackgroundService>();
我总结来说就是定义了一个后台服务类,用于处理请求队列。它依赖于 QueueTesting
类来处理请求队列的操作,然后在具体的耗时方法哪里把耗时方法添加到队列让他不影响整体运行时长。