【aspnetcore】用ConcurrentQueue实现一个简单的队列系统
第一步:定义队列服务接口
public interface ISimpleQueueServer
{
/// <summary>
/// 添加队列消息
/// </summary>
/// <param name="message">消息</param>
/// <param name="clientName">客户端名称</param>
/// <returns></returns>
string Add(string message, string clientName);
}
第二步:添加队列服务接口的实现
public class SimpleQueueServer : ISimpleQueueServer
{
/// <summary>
/// 队列
/// </summary>
private static ConcurrentQueue<string> _queue = new ConcurrentQueue<string>(); /// <summary>
/// 日志
/// </summary>
private static ILogger _log; /// <summary>
/// 后台任务
/// </summary>
private static Task _task; /// <summary>
/// 连续获取队列为空的次数
/// </summary>
private int EmptyRepeatCount = 0; /// <summary>
/// 属性,后台任务
/// </summary>
private Task MyTask
{
get
{
if (_task == null)
{
_task = new Task(MessageHandler);
}
return _task;
}
} /// <summary>
/// 构造函数
/// </summary>
/// <param name="factory"></param>
public SimpleQueueServer(ILoggerFactory factory)
{
if (_log == null)
{
_log = factory.CreateLogger("SimpleQueueServer");
} MyTask.Start();
} /// <summary>
/// 添加消息到队列
/// </summary>
/// <param name="message">消息</param>
/// <param name="clientName">发送的客户端名称</param>
/// <returns></returns>
public string Add(string message, string clientName = "")
{
try
{
string prefix = string.IsNullOrWhiteSpace(clientName) ? "" : $"【{clientName}】";
_queue.Enqueue($"{prefix}{message}");
return "OK";
}
catch (Exception ex)
{
_log.LogError(ex, "向队列添加信息失败");
return ex.Message;
}
} /// <summary>
/// 队列中要实现的任务
/// </summary>
/// <param name="threadName">线程名称,如果多</param>
/// <returns></returns>
private Action MessageHandler => () =>
{
while (true)
{
try
{
if (_queue.IsEmpty)
{
_log.LogDebug($"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} 队列为空");
Thread.Sleep(3000);
}
else
{
if (_queue.TryDequeue(out string result))
{
_log.LogDebug($"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} 获取到数据:{result}");
}
else
{
_log.LogDebug($"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} 尝试从队列获取消息失败");
}
Thread.Sleep(500);
}
}
catch (Exception ex)
{
_log.LogError(ex, "系统错误");
}
}
};
}
第三步:在startup中注册服务,这里稍稍装个x,定义一个IServiceCollection扩展,让代码看起来x格稍微高点
public static class ServiceCollectionExtension
{
public static void AddSimpleQueueServer(this IServiceCollection services)
{
services.AddSingleton<ISimpleQueueServer, SimpleQueueServer>((provider) =>
{
return new SimpleQueueServer(provider.GetService<ILoggerFactory>());
});
}
}
第四步:在startup的ConfigureServices中添加服务
services.AddSimpleQueueServer();
第五步:修改appsettings.Development.json文件
{
"Logging": {
"LogLevel": {
"Default": "Debug",
"System": "Warning",
"Microsoft": "Warning"
}
}
}
修改System和Microsoft的日志级别,防止调试时队列显示的消息淹没在无穷无尽的info中。
=======无聊的分割线======
新建一个QueueController
namespace AspnetCoreMvcStudy.Controllers
{
public class QueueController : Controller
{
private ISimpleQueueServer _server; public QueueController(ISimpleQueueServer server)
{
_server = server;
} public IActionResult Index()
{
return View();
} [HttpPost]
public JsonResult Send(string msg, string client)
{
for (int i = 0; i < 100; i++)
{
_server.Add($"{msg}-{i}", client);
}
return Json(new { Code = 200 });
}
}
}
创建视图 Index.cshtml
@{
ViewData["Title"] = "Index";
} <h2>队列测试</h2> <form id="form1" onsubmit="return false;">
<div class="form-group">
<label for="message">客户端名称</label>
<input type="text" id="client" value="" />
<label for="message">发送内容</label>
<input type="text" id="message" value="" />
<hr />
<button id="btnSubmit" class="btn btn-success">发送</button>
<span class="text-danger" id="info"></span>
</div>
</form> @section scripts
{
<script>
$('#btnSubmit').on('click', function () {
var that = $(this);
that.attr('disabled', 'disabled');
$.post('/Queue/Send', { msg: $('#message').val(), client: $('#client').val() }, function (response) {
if (response.code == 200) {
$('#info').text('发送成功');
} else {
$('#info').text(response.message);
}
that.removeAttr('disabled');
});
});
</script>
}
运行程序,开整
【aspnetcore】用ConcurrentQueue实现一个简单的队列系统的更多相关文章
- 《Linux内核分析》第三周 构建一个简单的Linux系统MenuOS
[刘蔚然 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000] WEEK THREE ...
- Linux第三周学习总结——构造一个简单的Linux系统MenuOS
第三周学习总结--构造一个简单的Linux系统MenuOS 作者:刘浩晨 [原创作品转载请注明出处] <Linux内核分析>MOOC课程http://mooc.study.163.com/ ...
- Python高级编程之生成器(Generator)与coroutine(四):一个简单的多任务系统
啊,终于要把这一个系列写完整了,好高兴啊 在前面的三篇文章中介绍了Python的Python的Generator和coroutine(协程)相关的编程技术,接下来这篇文章会用Python的corout ...
- Linux内核设计第三周——构造一个简单的Linux系统
Linux内核设计第三周 ——构造一个简单的Linux系统 一.知识点总结 计算机三个法宝: 存储程序计算机 函数调用堆栈 中断 操作系统两把宝剑: 中断上下文的切换 进程上下文的切换 linux内核 ...
- 第三节 构造一个简单的Linux系统MenuOS——20135203齐岳
第三节 构造一个简单的Linux系统MenuOS By 20135203齐岳 Linux内核源代码 arch/ 支持不同cpu的源代码 Documentations/ 文档存储 init/ 内核启动相 ...
- Linux内核分析第三周学习总结:构造一个简单的Linux系统MenuOS
韩玉琪 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.Linux内 ...
- 一个简单的CS系统打包过程图文版
一个简单的CS系统打包过程图文版 1. 打包内容 1.1. 此次打包的要求和特点 主工程是一个CS系统: 此CS系统运行的先决条件是要有.Net Framework 3.5: 主工程安装完成 ...
- Linux下一个简单的日志系统的设计及其C代码实现
1.概述 在大型软件系统中,为了监测软件运行状况及排查软件故障,一般都会要求软件程序在运行的过程中产生日志文件.在日志文件中存放程序流程中的一些重要信息, 包括:变量名称及其值.消息结构定义.函数返回 ...
- python制作一个简单的中奖系统
注释: 展示图下的代码,我是用pycharm写的,是python解释器中的一种,本课没不同解释器的要求,可根据自己喜欢的解释器编写. 步骤: 本期给大家带来的是,一个简单的中奖系统,首先打开自己电脑上 ...
随机推荐
- jquery特效(5)—轮播图③(鼠标悬浮停止轮播)
今天很无聊,就接着写轮播图了,需要说明一下,这次的轮播图是在上次随笔中jquery特效(3)—轮播图①(手动点击轮播)和jquery特效(4)—轮播图②(定时自动轮播)的基础上写出来的,也就是本次随笔 ...
- mysql优化20条原则
今天,数据库的操作越来越成为整个应用的性能瓶颈了,这点对于Web应用尤其明显.关于数据库的性能,这并不只是DBA才需要担心的事,而这更是我们程序员需要去关注的事情.当我们去设计数据库表结构,对操作数据 ...
- html5--3.22 综合实例03
html5--3.22 综合实例03 学习要点 通过一个综合实例来回顾学过的内容 这一章的内容比较多,不必强求一下子全记住,多做一些练习,用得多了自然就可以记住了 可以自己找一些实例练练手,比如各网站 ...
- 运行swoole_server方法
运行 php 文件 server.php 运行结果是如下: 只是服务器开启了 如果想看客户端连接的情况 可以测试一下 从新连接一个连接 用命令 方式 telnet 127.0.0.1 9501 这个9 ...
- C和C++语言&
#include "stdafx.h"#include "iostream"#include "animal.h"using namespa ...
- Java笔记(八)
GUI: awt和swing: java.awt:Abstract Window ToolKit(抽象窗口工具包),需要调用本地系统方法实现功能.属于重量级控件. javax.swing:在awt的基 ...
- NMS 原理 了解
NMS 原理:对于Bounding Box的列表B及其对应的置信度S,采用下面的计算方式.选择具有最大score的检测框M,将其从B集合中移除并加入到最终的检测结果D中.通常将B中剩余检测框中与M的I ...
- C/C++的const区别
1.const基础知识(用法.含义.好处) int main() { const int a; //a为const,常数型数 int const b; //b为const,常数型数 const int ...
- Java原子属性更新器AtomicReferenceFieldUpdater的使用
AtomicReferenceFieldUpdater是基于反射的工具类,用来将指定类型的指定的volatile引用字段进行原子更新,对应的原子引用字段不能是private的.通常一个类volatil ...
- MTK OTG 流程
一.注册mt_usb驱动 kernel-3.18/drivers/misc/mediatek/usb20/mt6735/usb20.c static int __init usb20_init(voi ...