基于 abp vNext 和 .NET Core 开发博客项目 - 集成Hangfire实现定时任务处理

上一篇文章成功使用了Redis缓存数据,大大提高博客的响应性能。

接下来,将完成一个任务调度中心,关于定时任务有多种处理方式,如果你的需求比较简单,比如就是单纯的过多少时间循环执行某个操作,可以直接使用.net core中内置的实现方式,新建一个类继承BackgroundService,实现ExecuteAsync()既可。

看一个例子,我们每过一秒输出一句HelloWorld,并写入日志中。

.BackgroundJobs中新建一个Jobs文件夹,添加HelloWorldJob.cs,并且继承自BackgroundService

//HelloWorldJob.cs
using log4net;
using Microsoft.Extensions.Hosting;
using System;
using System.Threading;
using System.Threading.Tasks;namespace Meowv.Blog.BackgroundJobs.Jobs
{public class HelloWorldJob : BackgroundService{private readonly ILog _log;public HelloWorldJob(){_log = LogManager.GetLogger(typeof(HelloWorldJob));}protected override async Task ExecuteAsync(CancellationToken stoppingToken){while (!stoppingToken.IsCancellationRequested){var msg = $"CurrentTime:{ DateTime.Now}, Hello World!";Console.WriteLine(msg);_log.Info(msg);await Task.Delay(1000, stoppingToken);}}}
}

然后在.HttpApi.Hosting层模块类中的ConfigureServices()注入context.Services.AddTransient<IHostedService, HelloWorldJob>();使用,运行一下看看效果。

可以看到已经成功输出了,你可以在ExecuteAsync()中做你的事件处理逻辑。这应该是最简单后台定时任务处理了,比较单一。

在abp框架中,官方给我们提供了许多后台工作的集成方式,有兴趣的可以自行研究一下,文档地址:https://docs.abp.io/zh-Hans/abp/latest/Background-Jobs

在本项目中,我将使用 Hangfire 来完成定时任务处理,为什么选择它呢?因为简单,开箱即用。下面进入正题,可以先将 HelloWorldJob 停掉。

.BackgroundJobs中添加nuget包:Volo.Abp.BackgroundJobs.HangFireHangfire.MySql.CoreHangfire.Dashboard.BasicAuthorizationVolo.Abp.AspNetCore,然后添加项目引用:.Domain

在根目录新建模块类:MeowvBlogBackgroundJobsModule.cs,继承AbpModule,依赖AbpBackgroundJobsHangfireModule

//MeowvBlogBackgroundJobsModule.cs
using Hangfire;
using Hangfire.MySql.Core;
using Meowv.Blog.Domain.Configurations;
using Meowv.Blog.Domain.Shared;
using Volo.Abp;
using Volo.Abp.BackgroundJobs.Hangfire;
using Volo.Abp.Modularity;namespace Meowv.Blog.BackgroundJobs
{[DependsOn(typeof(AbpBackgroundJobsHangfireModule))]public class MeowvBlogBackgroundJobsModule : AbpModule{public override void ConfigureServices(ServiceConfigurationContext context){context.Services.AddHangfire(config =>{config.UseStorage(new MySqlStorage(AppSettings.ConnectionStrings,new MySqlStorageOptions{TablePrefix = MeowvBlogConsts.DbTablePrefix + "hangfire"}));});}public override void OnApplicationInitialization(ApplicationInitializationContext context){var app = context.GetApplicationBuilder();app.UseHangfireServer();app.UseHangfireDashboard();}}
}

ConfigureServices()中添加配置,因为之前选用了MySQL,所以这里引用了Hangfire.MySql.Core这个包,相对于的其它数据库可以在nuget上寻找。

new MySqlStorage()中配置连接字符串,new MySqlStorageOptions()中配置表前缀,Hangfire会在第一次运行时,自动为我们创建表。

然后在OnApplicationInitialization()中进行使用,app.UseHangfireServer()必须调用,如果你不需要界面显示可以不用app.UseHangfireDashboard();

最后不要忘记,在.HttpApi.Hosting层模块类中依赖定时任务模块MeowvBlogBackgroundJobsModule

现在运行一下项目,打开地址:.../hangfire 看看。

数据库默认已经为我们创建了hangfire所需的表。

有一个地方要注意,就是在连接字符串中需要开启用户变量,修改一下appsettings.json中的连接字符串,在末尾添加:Allow User Variables=True

同时在app.UseHangfireDashboard()中,还支持很多配置项,现在我们这个定时任务是公开的,如果我们不想要外人访问,可以开启BasicAuth。

现在配置文件中配置Hangfire的登录账号和密码。

...
"Hangfire": {"Login": "meowv","Password": "123456"
}
...
...
/// <summary>
/// Hangfire
/// </summary>
public static class Hangfire
{public static string Login => _config["Hangfire:Login"];public static string Password => _config["Hangfire:Password"];
}
...

开启方式也很简单,之前已经引用了Hangfire.Dashboard.BasicAuthorization这个包,直接看代码。

app.UseHangfireDashboard(options: new DashboardOptions
{Authorization = new[]{new BasicAuthAuthorizationFilter(new BasicAuthAuthorizationFilterOptions{RequireSsl = false,SslRedirect = false,LoginCaseSensitive = true,Users = new []{new BasicAuthAuthorizationUser{Login = AppSettings.Hangfire.Login,PasswordClear =  AppSettings.Hangfire.Password}}})},DashboardTitle = "任务调度中心"
});

app.UseHangfireDashboard()中可以自定义访问路径,我们这里没有传,就是用默认值。自定义界面的标题Title等等。更多参数可以自己看DashboardOptions,结合情况来使用,编译运行看看效果。

现在就需要输入我们配置的账号密码才可以进入Hangfire界面了。

这样我们就集成好了Hangfire,并且还有了一个可视化的界面,接下来我们同样实现一个简单的定时任务看看效果。

在Jobs文件夹添加一个接口:IBackgroundJob,让他继承ITransientDependency,实现依赖注入,同时定义一个方法ExecuteAsync()

//IBackgroundJob.cs
using System.Threading.Tasks;
using Volo.Abp.DependencyInjection;namespace Meowv.Blog.BackgroundJobs.Jobs
{public interface IBackgroundJob : ITransientDependency{/// <summary>/// 执行任务/// </summary>/// <returns></returns>Task ExecuteAsync();}
}

在Jobs文件夹新建文件夹Hangfire,添加HangfireTestJob.cs,继承IBackgroundJob实现ExecuteAsync()方法。

//HangfireTestJob.cs
using System;
using System.Threading.Tasks;namespace Meowv.Blog.BackgroundJobs.Jobs.Hangfire
{public class HangfireTestJob : IBackgroundJob{public async Task ExecuteAsync(){Console.WriteLine("定时任务测试");await Task.CompletedTask;}}
}

这样就完成了定时任务的逻辑,我们怎么来调用呢?新建一个扩展方法MeowvBlogBackgroundJobsExtensions.cs

//MeowvBlogBackgroundJobsExtensions.cs
using Hangfire;
using Meowv.Blog.BackgroundJobs.Jobs.Hangfire;
using Microsoft.Extensions.DependencyInjection;
using System;namespace Meowv.Blog.BackgroundJobs
{public static class MeowvBlogBackgroundJobsExtensions{public static void UseHangfireTest(this IServiceProvider service){var job = service.GetService<HangfireTestJob>();RecurringJob.AddOrUpdate("定时任务测试", () => job.ExecuteAsync(), CronType.Minute());}}
}

这里使用IServiceProvider解析服务,获取到我们的实列,所以我们可以在模块类中的OnApplicationInitialization(...)中直接调用此扩展方法。

RecurringJob.AddOrUpdate()是定期作业按指定的计划触发任务,同时还有EnqueueScheduleContinueJobWith等等,可以看一下Hangfire官方文档:https://docs.hangfire.io/en/latest/

CronType是自定义的一个静态类,他帮我们自动生成了Cron表达式,这里表示一分钟执行一次,关于不懂Cron的同学,可以去自学一下,也许看看下面代码就懂了,也有许多Cron表达式在线生成的工具。

# Example of job definition:
# .---------------- minute (0 - 59)
# |  .------------- hour (0 - 23)
# |  |  .---------- day of month (1 - 31)
# |  |  |  .------- month (1 - 12) OR jan,feb,mar,apr ...
# |  |  |  |  .---- day of week (0 - 6) (Sunday=0 or 7) OR sun,mon,tue,wed,thu,fri,sat
# |  |  |  |  |
*/30 * * * * /bin/python /qix/spider/spider.py #每30分钟执行一次

直接在根目录添加MeowvBlogCronType.cs

//MeowvBlogCronType.cs
using Hangfire;
using System;namespace Meowv.Blog.BackgroundJobs
{/// <summary>/// Cron类型/// </summary>public static class CronType{/// <summary>/// 周期性为分钟的任务/// </summary>/// <param name="interval">执行周期的间隔,默认为每分钟一次</param>/// <returns></returns>public static string Minute(int interval = 1){return "1 0/" + interval.ToString() + " * * * ? ";}/// <summary>/// 周期性为小时的任务/// </summary>/// <param name="minute">第几分钟开始,默认为第一分钟</param>/// <param name="interval">执行周期的间隔,默认为每小时一次</param>/// <returns></returns>public static string Hour(int minute = 1, int interval = 1){return "1 " + minute + " 0/" + interval.ToString() + " * * ? ";}/// <summary>/// 周期性为天的任务/// </summary>/// <param name="hour">第几小时开始,默认从1点开始</param>/// <param name="minute">第几分钟开始,默认从第1分钟开始</param>/// <param name="interval">执行周期的间隔,默认为每天一次</param>/// <returns></returns>public static string Day(int hour = 1, int minute = 1, int interval = 1){return "1 " + minute.ToString() + " " + hour.ToString() + " 1/" + interval.ToString() + " * ? ";}/// <summary>/// 周期性为周的任务/// </summary>/// <param name="dayOfWeek">星期几开始,默认从星期一点开始</param>/// <param name="hour">第几小时开始,默认从1点开始</param>/// <param name="minute">第几分钟开始,默认从第1分钟开始</param>/// <returns></returns>public static string Week(DayOfWeek dayOfWeek = DayOfWeek.Monday, int hour = 1, int minute = 1){return Cron.Weekly(dayOfWeek, hour, minute);}/// <summary>/// 周期性为月的任务/// </summary>/// <param name="day">几号开始,默认从一号开始</param>/// <param name="hour">第几小时开始,默认从1点开始</param>/// <param name="minute">第几分钟开始,默认从第1分钟开始</param>/// <returns></returns>public static string Month(int day = 1, int hour = 1, int minute = 1){return Cron.Monthly(day, hour, minute);}/// <summary>/// 周期性为年的任务/// </summary>/// <param name="month">几月开始,默认从一月开始</param>/// <param name="day">几号开始,默认从一号开始</param>/// <param name="hour">第几小时开始,默认从1点开始</param>/// <param name="minute">第几分钟开始,默认从第1分钟开始</param>/// <returns></returns>public static string Year(int month = 1, int day = 1, int hour = 1, int minute = 1){return Cron.Yearly(month, day, hour, minute);}}
}

接着就可以调用定时任务了。

//MeowvBlogBackgroundJobsModule.cs
...
public override void OnApplicationInitialization(ApplicationInitializationContext context)
{var app = context.GetApplicationBuilder();...var service = context.ServiceProvider;service.UseHangfireTest();
}
...

通过context.ServiceProvider可以获取到IServiceProvider,然后直接调用扩展方法,是不是超级简单,现在编译运行项目看效果。

关于任务是否真的运行成功,我们可以从输出看出。

完美,本篇完成了Hangfire的集成,并实现了一个定时任务计划,有没有发现很简单,你学会了吗?????????????

开源地址:https://github.com/Meowv/Blog/tree/blog_tutorial

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

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

相关文章

Docker基本组成 和 基本命令

此篇为Docker笔记&#xff0c;文章可能存在疏忽&#xff0c;建议直接观看原视频。 视频地址&#xff1a;https://www.bilibili.com/video/BV1og4y1q7M4?spm_id_from333.999.0.0 Docker基本组成 和 基本命令 镜像 image&#xff1a;就好比一个模板&#xff0c;可以通过这个模板…

Docker镜像讲解

此篇为Docker笔记&#xff0c;文章可能存在疏忽&#xff0c;建议直接观看原视频。 视频地址&#xff1a;https://www.bilibili.com/video/BV1og4y1q7M4?spm_id_from333.999.0.0 参考&#xff1a;https://blog.csdn.net/11b202/article/details/21389067 Docker镜像讲解 镜像是…

Making the Grade POJ - 3666(离散化+dp)

题意&#xff1a; 给你n个山的高度&#xff0c;单独的一个数可以任意加减&#xff0c;让经过对每座山峰任意加减高度后变成递增或递减的序列时&#xff0c;求对每个数的相加或相减的数目的最小和。 题目&#xff1a; A straight dirt road connects two fields on FJ’s far…

Kubernetes的安全性怎么解?从4个方面为你列出方案清单

导语Kubernetes中的安全性是一个多维问题&#xff0c;必须从各个不同的角度来解决才算完善&#xff0c;这篇文章将从4个方面为读者列出安全清单。正文Kubernetes&#xff0c;经过更快的采用和社区的更多贡献&#xff0c;正日益攀登到新的高度。不过&#xff0c;安全性仍然是Kub…

DDIA笔记—第六章 数据分区

第六章 数据分区 数据分区与数据复制 分区通常与复制结合使用&#xff0c;即每个分区在多个节点都存在副本&#xff0c;这就意味着某条记录属于特定的分区&#xff0c;而同样的内容会保存在不同的节点上以提高系统的容错性。 每个节点同时充当某些分区的主副本和其他分区的从…

Magicodes.IE 2.2发布

Magicodes.IE 2.2发布导入导出通用库&#xff0c;支持DTO导入导出以及动态导出&#xff0c;支持Excel、Word、PDF、CSV和HTML。已加入ncc开源组织.Magicodes.IE2.0发布Magicodes.IE2.1发布如何做好一个开源项目(一)GitHub&#xff1a;https://github.com/dotnetcore/Magicodes.…

C++ 基类,子对象,派生类构造函数调用顺序

#include <iostream> using namespace std;class A {public:A( ) {cout << "A Constructor………" << endl;}~A( ) {cout << "A Destructor………" << endl;} };class B: public A {public:B( ) {cout << "B …

C++ 虚析构函数

代码如下: #include <iostream> using namespace std;class Base {public:Base() {cout << "Base" << endl;}~Base() {cout << "Base destructor" << endl;} };class Derived : public Base {public:Derived() {cout <&…

I - Interesting Permutation Gym - 102394I(排列组合)

题意&#xff1a; 纯数题 1≤i≤n, fimax{a1,a2,…,ai}; 1≤i≤n, gimin{a1,a2,…,ai}; 1≤i≤n, hifi−gi. 数列a是一个排列&#xff0c;问多少种排列方式满足h数列。 题目&#xff1a; DreamGrid has an interesting permutation of 1,2,…,n denoted by a1,a2,…,an. He …

Magicodes.SwaggerUI 已支持.NET Core 3.1

Magicodes.SwaggerUI 通过配置文件简单配置即可快速完成SwaggerUI的配置&#xff0c;包括&#xff1a;SwaggerUI的文档信息API分组API隐藏API JSON生成&#xff08;枚举、API架构Id&#xff09;验证自定义页面支持.NET Core 2.2和3.1。版本日志和使用教程见下文。注意&#xff…

[推荐]大量 Blazor 学习资源(二)

继上一篇《[推荐]大量 Blazor 学习资源&#xff08;一&#xff09;》之后&#xff0c;社区反应不错&#xff0c;但因个人原因导致这篇文章姗姗来迟&#xff0c;不过最终还是来了&#xff01;这篇文章主要收集一些常用组件、书籍和电子书。资料来源&#xff1a;https://github.c…

Sql Server之旅——第八站 看公司这些DBA们设计的这些复合索引

这一篇再说下索引的最后一个主题&#xff0c;索引覆盖&#xff0c;当然学习比较好的捷径是看看那些大师们设计的索引&#xff0c;看从中能提取些什么营养的东西&#xff0c;下面我们看看数据库中一个核心的Orders表。一&#xff1a;查看表的架构1. 先查看这个表的大概架构信息-…

C++ setprecision()用法

io 流控制头文件, 主要是一些操纵用法如setw(int n),setprecision(int n) #include < iomanip > setw(n)用法&#xff1a; 通俗地讲就是预设宽度 #include<iostream> #include <iomanip> using namespace std;int main() {cout << setw(5) <<…

备战ccpc分站赛:秦皇岛和威海站(数论模块和dp模块)

挑战程序设计竞赛&#xff08;第2版&#xff09;练习题 tips&#xff1a;难度&#xff08;个人主观判断&#xff09;&#xff1a; 简单* 简单但卡思维 ** 中 *** 中稍加思考 **** 难 ***** 1 . 记录结果再利用的“动态规划” &#xff08;1&#xff09;基础的动态规划算法&am…

15分钟从零开始搭建支持10w+用户的生产环境(四)

上一篇文章&#xff0c;介绍了这个架构中&#xff0c;WebServer的选择&#xff0c;以及整个架构中扩展时的思路。原文地址&#xff1a;15分钟从零开始搭建支持10w用户的生产环境(三)五、架构实践前边用了三篇文章&#xff0c;详细介绍了这个架构的各个部分的选择以及安装。这篇…

[Java基础]体验Stream流

代码如下: package StreamTest;import java.lang.reflect.Array; import java.util.ArrayList;public class StreamDemo {public static void main(String[] args){ArrayList<String> list new ArrayList<String>();list.add("Tom");list.add("ja…

Cow Bowling POJ - 3176(基础的动态规划算法)

题意&#xff1a; 杨辉三角&#xff0c;让从顶部开始走到底部&#xff0c;所经过的每一层的点数相加&#xff0c;使得实现最高和。 题目&#xff1a; The cows don’t use actual bowling balls when they go bowling. They each take a number (in the range 0…99), thoug…

[Java基础]Stream流的常见生成方式

1.Collection体系的集合可以使用默认方法stream()生成流 default Stream< E > stream() 代码如下: package StreamTest;import java.util.ArrayList; import java.util.HashSet; import java.util.List; import java.util.Set; import java.util.stream.Stream;public …