基于C#的计时管理器

问题

我们使用各种系统时候会遇到以下问题:

  • 12306上购买火车票如果15分钟内未完成支付则订单自动取消。

  • 会议场馆预定座位,如果10分钟内未完成支付则预定自动取消。

  • 在指定时间之后,我需要执行一项任务。

我之前做的很多系统,往往都是定期执行一个特定任务。而上诉问题都涉及到滑动窗口时间的定时任务。

比如:我早上10点20分预定了一张火车票,我需要在15分钟内支付完成,否则订单会被取消。同一时间可能会有成百上千的人预定其他火车票,我需要在每个人的15分钟期限达时候执行检查,如果还未支付则自动取消订单。

方案

我们搞清楚了要解决的问题以后,我们来思考方案。有经验的程序员会立即思考出下面的方案:

  • 使用消息队列的延迟投送功能,每个订单添加成功后发送一个延迟15分钟的延迟消息。订单状态处理器15分钟后收到消息,检查支付状态,如果未支付则取消订单。

  • Redis也有类似的功能,原理大致相同。

但我不想使用消息队列的功能,因为延迟消息投送是一种技术实现,我希望用代码反应这种业务实现,所以用纯代码来处理他。(我并不是为了从新发明车轮,因为这是一种业务需求,会有变化扩展的需要,所以决定自己尝试做一下增加经验)

算法思路:

我们的需求定时时间都在15分钟以内,假定都没有超过1个小时的或者几天的。(如果超过1个小时的,可以扩展这个设计,这篇暂时不展开讨论)

我们可考虑将一个小时分成3600秒,每秒代表一个位置来存储所有到期的订单,当下单的时候根据当前时间 加上 15分钟时间间隔,我们就可以得到15分钟以后的时间,将这个订单添加到对应的位置上。

数据结构选择:

我们选择C#中提供的最新的并发字典作为基础数据结构,Key值是3600秒中的每一秒的数值,内容是一个队列用于存放该时间点的所有订单。

39208b8b769b0a8bbcb1833b8f485304.png

public ConcurrentDictionary<int, ConcurrentQueue<IJob>> jobs = new ConcurrentDictionary<int, ConcurrentQueue<IJob>>();

数据结构我们思考好了,其实功能就完成了大半了,代码的设计也就基本定下来了。

代码实现(我喜欢使用控制台应用程序做实验)
  1. 建立一个定时管理器

    public class TimerManager{//并发字典存储需要检查的任务(这里可以是订单检查任务,每个任务可以包含一个订单Id)public ConcurrentDictionary<int, ConcurrentQueue<IJob>> jobs = new ConcurrentDictionary<int, ConcurrentQueue<IJob>>();private Timer timer;public TimerManager(){//每间隔1秒钟执行一次。和当前时间同步。timer = new Timer(ProcessJobs, null, 0, 1000);}}
  2. 增加一个任务到字典

    /// <summary>/// 增加一个任务到时间字典中/// </summary>/// <param name="timeKey">根据延迟时间计算出的key值</param>/// <param name="duetime">毫秒单位</param>/// <exception cref="NotImplementedException"></exception>public void AddJob(IJob job, TimeSpan duetime){var key = GetKey(duetime);ConcurrentQueue<IJob> queue = new ConcurrentQueue<IJob>();queue.Enqueue(job);jobs.AddOrUpdate(key, queue, (key, jobs) =>{jobs.Enqueue(job);return jobs;});}
  3. 根据时间计算Key的方法

    /// <summary>/// 根据延迟时间生成当前键值/// </summary>/// <param name="duetime"></param>/// <returns></returns>private int GetKey(TimeSpan duetime){var currentDateTime = DateTime.Now;//到期时间var targetDateTime = currentDateTime.Add(duetime);//不要忘了把分钟换算成秒,然后在和延迟时间相加就得到Keyvar key = targetDateTime.Minute * 60 + targetDateTime.Second;return key;}
  4. 将任务添加到字典

    /// <summary>/// 增加一个任务到时间字典中/// </summary>/// <param name="job">需要执行的任务</param>/// <param name="duetime">多少时间间隔后检查</param>public void AddJob(IJob job, TimeSpan duetime){var key = GetKey(duetime);ConcurrentQueue<IJob> queue = new ConcurrentQueue<IJob>();queue.Enqueue(job);//这是并发字典的方法,这里就是当Key不存在就增加新的值进去,当Key存在就在Key的队列中增加一个新任务jobs.AddOrUpdate(key, queue, (key, jobs) =>{jobs.Enqueue(job);return jobs;});}
  5. 计时器每秒执行时处理任务的方法,循环从队列中取出任务直到所有任务处理完毕。

    private async void ProcessJobs(object state){//根据当前时间计算Key值var key = DateTime.Now.Minute * 60 + DateTime.Now.Second;Console.WriteLine(key);//查找Key值对应的任务队列并处理。bool keyExists = jobs.TryGetValue(key, out var jobQueue);if (keyExists){IJob job;while(jobQueue.TryDequeue(out job)){await job.Run();}}}
  6. 代码中设计IJob 和Job的一个实现,为了易于理解,这个job没有做太多事情。如果需要扩展去检查订单,可以在这里记录订单Id,创建任务的时候将订单ID和任务关联,这样定时器处理这个任务的时候能找到对应订单了。

    public interface IJob{Task Run();}/// <summary>/// 代表一个工作/// </summary>public class Job : IJob{public Guid JobId { get; set; }public Job(){JobId = Guid.NewGuid();}public async Task Run(){Console.WriteLine(" Job Id: " + JobId.ToString() + " is running.");await Task.Delay(2000);Console.WriteLine(" Job Id:" + JobId.ToString() + " have completed.");}}
  7. 主程序Programe中调用定时管理器

    using TimerTest;Console.WriteLine("Hello, World!");TimerManager timerManager = new TimerManager();Job job1 = new Job();// 添加一个任务1分钟后执行
    timerManager.AddJob( job1, TimeSpan.FromMinutes(1));// 在添加另一个任务2分钟后执行
    Job job2 = new Job();
    timerManager.AddJob(job2, TimeSpan.FromMinutes(2));Console.ReadLine();
执行结果

结果中可以看到, 任务1 在1014的键值上被处理,1014的键值对应的时间是 16:54 秒,也就是在我运行这个程序1分钟后。

3fe2079650c6fa711da7ed9ceeadd848.png

任务添加时间执行时间
第一次任务(计时1分钟)15:5416:54
第二次任务(计时2分钟)15:5417:54

任务2 在 1074的键值上被处理,1074对应的时间是 17:54 秒 执行。从上表可以看出程序正常运行得出结果。

3e0201cf9a0149a3c179a485f5a4c84a.png

总结

这是一个简单的控制台程序验证了这个定时管理器的实现方法,我们将1个小时分成3600秒,每一秒对应一个Key值,在这个值上我们存储需要被处理的任务。在增加任务时候,我们也用同样的算法确定这个Key值。处理的时候根据当前时间计算除Key值进行处理。

这样的话,在真实场景中,我们有3600个Key值可以存储每一秒钟用户提交的所有订单,时间没走过1秒我就处理对应的任务。

后续可以完善的地方
  • 我们可以将这个类添加到ASP.NET MVC中,使用依赖注入为单实例生命周期,并发字典和并发队列是线程安全的,所以这里可以放心使用。

  • 我们可以扩展Job方法,根据业务逻辑添加更多的信息以便于处理。例如处理订单的ID,或其他什么业务ID。

  • 处理任务的方法可以采用多个消费者并发执行,增加处理速度。

  • 可以将任务实体存储到数据库,以便于应对突发宕机事故可以快速重建任务。

  • 当然我们也可以用Hangfire来轻松实现这个业务。

    var jobId = BackgroundJob.Schedule(() => Console.WriteLine("Delayed!"),TimeSpan.FromDays(7)); //这里改成分钟就好了

最后祝.NET 20周年快乐。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/291523.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言之malloc和free总结

1、内存分配和函数介绍 内存分配:指在程序执行的过程空间中或者回收存储空间 ,内存分配分为静态内存分配和动态内存分配 ,要实现动态内存分配,就需要有执行这个操作的对象。C语言提供的两个标准库函数:malloc和free。 1)malloc函数 原型:void *malloc(size_t size)…

哈希表(hashtable)的javascript简单实现

javascript中没有像c#,java那样的哈希表(hashtable)的实现。在js中,object属性的实现就是hash表,因此只要在object上封装点方法,简单的使用obejct管理属性的方法就可以实现简单高效的hashtable。 首先简单的介绍关于属性的一些方法&#xff1a; 属性的枚举: for/in循环是遍历对…

HDU 2516 (Fabonacci Nim) 取石子游戏

这道题的结论就是&#xff0c;石子的个数为斐波那契数列某一项的时候&#xff0c;先手必败&#xff1b;否则&#xff0c;先手必胜。 结论很简单&#xff0c;但是证明却不是特别容易。找了好几篇博客&#xff0c;发现不一样的也就两篇&#xff0c;但是这两篇给的证明感觉证得不清…

access的ole对象换成mysql_ACCESS的Ole对象读取写入

Ole对象在Access中存储为二进制文件&#xff0c;读取的时候需要注意转换出的文件的编码格式1OleDbConnection OleConnnewOleDbConnection();2OleConn.ConnectionString"ProviderMicrosoft.Jet.OleDb.4.0;data sourceD:\WorkStation\Dialy_Sol\Dialy\Dialy.mdb";3OleD…

C++之delete常见错误总结

1、动态分配内存后释放了一次,再次释放 1)直接删除2次 int main() {int *a = new int(50);cout<<*a<<endl;delete a;delete a;return 0; } 2)另外一个指针指向分配的内存,然后把这个2个指针都删除 int* p1 = new int(50); int* p2 = p1; //p2和p1 现在指向同一…

ABP vNext微服务架构详细教程——分布式权限框架(上)

1简介ABP vNext框架本身提供了一套权限框架&#xff0c;其功能非常丰富&#xff0c;具体可参考官方文档&#xff1a;https://docs.abp.io/en/abp/latest/Authorization但是我们使用时会发现&#xff0c;对于正常的单体应用&#xff0c;ABP vNext框架提供的权限系统没有问题&…

前端每隔几秒发送一个请求

2019独角兽企业重金招聘Python工程师标准>>> <html><head><SCRIPT LANGUAGE"JavaScript"> var timer;//声明一个定时器 var count 0; function test() { //每隔500毫秒执行一次add()方法 timer window.setInterval("add()"…

Android之走手机流量让电脑能上网几种方法

1、通过“USB共享网络"来使电脑上网 1)我是vivo手机&#xff0c;把手机插上电脑,打开usb调试&#xff0c;然后正常连接电脑 2&#xff09;在“设置”里面打开“个人热点”里面的“通过usb共享网络”开关 3&#xff09;切换网络连接&#xff0c;对比之前的没插上手机之前没…

element 表单回显验证_关于vue el-form表单报错的问题

在写el-form表单的时候&#xff0c;遇到了蛮多问题&#xff0c;在这里记录一下。1.表单验证报错[Element Warn][Form]model is required for validate to work!初始代码如下&#xff1a;<!-- 表单部分 --> <el-formref"inputForm"size"mini"inlin…

Objective-C NSSetNSMutableSet以及CountedSet

NSSet说实话,对这个类的应用,还不知道到底什么时候会用到,先过一遍脑子吧,以后有需要用到的时候,不至于陌生! #import <Foundation/Foundation.h> int main(int argc, const char * argv[]) { autoreleasepool { //创建4个NSNumber对象 NSNumber *obj1 [NSNumber number…

IOZONE测试工具使用方法(转载)

IOZONE主要用来测试操作系统文件系统性能的测试工具&#xff0c;该工具所测试的范围主要有&#xff0c;write , Re-write, Read, Re-Read, Random Read, Random Write, Random Mix, Backwards Read, Record Rewrite, Strided Read, Fwrite, Frewrite, Fread, Freread, Mmap, As…

如何通过 C# 判断某个 IP 所属的地区?

咨询区 RC1140如何通过 C# 判断某个 IP 所属的地区&#xff1f;这样我就可以方便统计。回答区 Jaimes可以借助第三方API接口&#xff0c;参考网址&#xff1a;https://ipapi.co/8.8.8.8/country/ &#xff0c; C# 代码如下&#xff1a;using System; using System.Net; using S…

4月12日 webform基本控件

服务器基本控件&#xff1a; button: text属性 linkbutton:text属性&#xff0c;它是一个超链接模样的普通button hyperlink: navigateurl:链接地址&#xff0c;相当于<a>标签 imagebutton:imageurl:指定图片路径&#xff0c;这也是一个按钮&#xff0c;执行click事件 im…

C/C++之函数返回值为指针或者是引用时常见错误总结

1、说明 函数如果是指针或则引用的返回,一般全局变量、局部静态变量、局部动态分配内存的变量可以使用作为函数的返回值,局部变量不行,因为局部变量函数调用完会自动销毁内存,这个时候返回的指针或则引用就有问题了。 2、展示代码 #include <iostream> #include <…

我做了一个 Istio Workshop,这是第一讲介绍

我是 Jimmy Song[1]&#xff0c;Tetrate 布道师&#xff0c;云原生社区创始人。你可以能想到为什么在这个时候创建一个 Istio 教程&#xff0c;因为市面上已经林林总总有不少关于 Istio 的书籍和教程了&#xff0c;但是我们都知道 Istio 是一个新兴技术&#xff0c;发展十分迅速…

Swoole入门指南:PHP7安装Swoole详细教程(一)

好久未更新了&#xff0c;不是懒呃&#xff0c;是太忙啦&#xff01;终于偷得浮生几日闲。这一段时间准备为大家带来swoole的入门教程&#xff0c;感受一下php的nodeJs强悍之处。 所有的示例代码均放在了github上&#xff1a;learn-swoole 环境 这里不在使用apache做为web serv…

mysql如何快速插入一千万条数据_如何快速安全的插入千万条数据?

最近有个需求解析一个订单文件&#xff0c;并且说明文件可达到千万条数据&#xff0c;每条数据大概在20个字段左右&#xff0c;每个字段使用逗号分隔&#xff0c;需要尽量在半小时内入库。思路1.估算文件大小因为告诉文件有千万条&#xff0c;同时每条记录大概在20个字段左右&a…

解决 同时安装 python3,python2环境时,用pip安装 python3 包

应用场景 默认mac上已经安装了 python2; 而我又安装了 python3&#xff0c;并使用 python3; 安装了 pip 默认&#xff0c;pip安装的包安装在了 python2上了&#xff1b; 但是我想用 pip把安装的包安装在 python3上 &#xff0c;所以如下解决方式&#xff1b; 1&#xff1a;在ma…

C/C++之#ifdef、#if、#if defined的区别

1、看代码 2、运行结果 3、分析 #fi&#xff1a;后面接的表达式&#xff0c;如果为1就编译包含里面的内容 #ifdef&#xff1a;后面接的是一个宏&#xff0c;只要定义这个宏就行 #if defined(x)&#xff1a;和#ifdef效果一样 #if !defined(x)&#xff1a;和#ifndef效果一样

如何下载EP的各个版本?

到teamCity上面去下载 http://adc00cbv.us.oracle.com:8090/ 这里面刚进去是什么都没有的&#xff0c;要点击 Configure visible projects 配置一下才会显示 EP的各个版本是在V6.1.1.X中去下载的&#xff0c;也即EP和prodika是在一起出release 版本的。 转载于:https://www.cnb…