上一篇文章(https://www.cnblogs.com/meowv/p/12956696.html)成功使用了Redis缓存数据,大大提高博客的响应性能。

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

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

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

  1. //HelloWorldJob.cs
  2. using log4net;
  3. using Microsoft.Extensions.Hosting;
  4. using System;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. namespace Meowv.Blog.BackgroundJobs.Jobs
  8. {
  9. public class HelloWorldJob : BackgroundService
  10. {
  11. private readonly ILog _log;
  12. public HelloWorldJob()
  13. {
  14. _log = LogManager.GetLogger(typeof(HelloWorldJob));
  15. }
  16. protected override async Task ExecuteAsync(CancellationToken stoppingToken)
  17. {
  18. while (!stoppingToken.IsCancellationRequested)
  19. {
  20. var msg = $"CurrentTime:{ DateTime.Now}, Hello World!";
  21. Console.WriteLine(msg);
  22. _log.Info(msg);
  23. await Task.Delay(1000, stoppingToken);
  24. }
  25. }
  26. }
  27. }

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

0

可以看到已经成功输出了,你可以在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

  1. //MeowvBlogBackgroundJobsModule.cs
  2. using Hangfire;
  3. using Hangfire.MySql.Core;
  4. using Meowv.Blog.Domain.Configurations;
  5. using Meowv.Blog.Domain.Shared;
  6. using Volo.Abp;
  7. using Volo.Abp.BackgroundJobs.Hangfire;
  8. using Volo.Abp.Modularity;
  9. namespace Meowv.Blog.BackgroundJobs
  10. {
  11. [DependsOn(typeof(AbpBackgroundJobsHangfireModule))]
  12. public class MeowvBlogBackgroundJobsModule : AbpModule
  13. {
  14. public override void ConfigureServices(ServiceConfigurationContext context)
  15. {
  16. context.Services.AddHangfire(config =>
  17. {
  18. config.UseStorage(
  19. new MySqlStorage(AppSettings.ConnectionStrings,
  20. new MySqlStorageOptions
  21. {
  22. TablePrefix = MeowvBlogConsts.DbTablePrefix + "hangfire"
  23. }));
  24. });
  25. }
  26. public override void OnApplicationInitialization(ApplicationInitializationContext context)
  27. {
  28. var app = context.GetApplicationBuilder();
  29. app.UseHangfireServer();
  30. app.UseHangfireDashboard();
  31. }
  32. }
  33. }

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

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

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

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

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

2

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

3

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

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

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

  1. ...
  2. "Hangfire": {
  3. "Login": "meowv",
  4. "Password": "123456"
  5. }
  6. ...
  1. ...
  2. /// <summary>
  3. /// Hangfire
  4. /// </summary>
  5. public static class Hangfire
  6. {
  7. public static string Login => _config["Hangfire:Login"];
  8. public static string Password => _config["Hangfire:Password"];
  9. }
  10. ...

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

  1. app.UseHangfireDashboard(options: new DashboardOptions
  2. {
  3. Authorization = new[]
  4. {
  5. new BasicAuthAuthorizationFilter(new BasicAuthAuthorizationFilterOptions
  6. {
  7. RequireSsl = false,
  8. SslRedirect = false,
  9. LoginCaseSensitive = true,
  10. Users = new []
  11. {
  12. new BasicAuthAuthorizationUser
  13. {
  14. Login = AppSettings.Hangfire.Login,
  15. PasswordClear = AppSettings.Hangfire.Password
  16. }
  17. }
  18. })
  19. },
  20. DashboardTitle = "任务调度中心"
  21. });

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

4

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

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

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

  1. //IBackgroundJob.cs
  2. using System.Threading.Tasks;
  3. using Volo.Abp.DependencyInjection;
  4. namespace Meowv.Blog.BackgroundJobs.Jobs
  5. {
  6. public interface IBackgroundJob : ITransientDependency
  7. {
  8. /// <summary>
  9. /// 执行任务
  10. /// </summary>
  11. /// <returns></returns>
  12. Task ExecuteAsync();
  13. }
  14. }

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

  1. //HangfireTestJob.cs
  2. using System;
  3. using System.Threading.Tasks;
  4. namespace Meowv.Blog.BackgroundJobs.Jobs.Hangfire
  5. {
  6. public class HangfireTestJob : IBackgroundJob
  7. {
  8. public async Task ExecuteAsync()
  9. {
  10. Console.WriteLine("定时任务测试");
  11. await Task.CompletedTask;
  12. }
  13. }
  14. }

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

  1. //MeowvBlogBackgroundJobsExtensions.cs
  2. using Hangfire;
  3. using Meowv.Blog.BackgroundJobs.Jobs.Hangfire;
  4. using Microsoft.Extensions.DependencyInjection;
  5. using System;
  6. namespace Meowv.Blog.BackgroundJobs
  7. {
  8. public static class MeowvBlogBackgroundJobsExtensions
  9. {
  10. public static void UseHangfireTest(this IServiceProvider service)
  11. {
  12. var job = service.GetService<HangfireTestJob>();
  13. RecurringJob.AddOrUpdate("定时任务测试", () => job.ExecuteAsync(), CronType.Minute());
  14. }
  15. }
  16. }

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

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

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

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

直接在根目录添加MeowvBlogCronType.cs

  1. //MeowvBlogCronType.cs
  2. using Hangfire;
  3. using System;
  4. namespace Meowv.Blog.BackgroundJobs
  5. {
  6. /// <summary>
  7. /// Cron类型
  8. /// </summary>
  9. public static class CronType
  10. {
  11. /// <summary>
  12. /// 周期性为分钟的任务
  13. /// </summary>
  14. /// <param name="interval">执行周期的间隔,默认为每分钟一次</param>
  15. /// <returns></returns>
  16. public static string Minute(int interval = 1)
  17. {
  18. return "1 0/" + interval.ToString() + " * * * ? ";
  19. }
  20. /// <summary>
  21. /// 周期性为小时的任务
  22. /// </summary>
  23. /// <param name="minute">第几分钟开始,默认为第一分钟</param>
  24. /// <param name="interval">执行周期的间隔,默认为每小时一次</param>
  25. /// <returns></returns>
  26. public static string Hour(int minute = 1, int interval = 1)
  27. {
  28. return "1 " + minute + " 0/" + interval.ToString() + " * * ? ";
  29. }
  30. /// <summary>
  31. /// 周期性为天的任务
  32. /// </summary>
  33. /// <param name="hour">第几小时开始,默认从1点开始</param>
  34. /// <param name="minute">第几分钟开始,默认从第1分钟开始</param>
  35. /// <param name="interval">执行周期的间隔,默认为每天一次</param>
  36. /// <returns></returns>
  37. public static string Day(int hour = 1, int minute = 1, int interval = 1)
  38. {
  39. return "1 " + minute.ToString() + " " + hour.ToString() + " 1/" + interval.ToString() + " * ? ";
  40. }
  41. /// <summary>
  42. /// 周期性为周的任务
  43. /// </summary>
  44. /// <param name="dayOfWeek">星期几开始,默认从星期一点开始</param>
  45. /// <param name="hour">第几小时开始,默认从1点开始</param>
  46. /// <param name="minute">第几分钟开始,默认从第1分钟开始</param>
  47. /// <returns></returns>
  48. public static string Week(DayOfWeek dayOfWeek = DayOfWeek.Monday, int hour = 1, int minute = 1)
  49. {
  50. return Cron.Weekly(dayOfWeek, hour, minute);
  51. }
  52. /// <summary>
  53. /// 周期性为月的任务
  54. /// </summary>
  55. /// <param name="day">几号开始,默认从一号开始</param>
  56. /// <param name="hour">第几小时开始,默认从1点开始</param>
  57. /// <param name="minute">第几分钟开始,默认从第1分钟开始</param>
  58. /// <returns></returns>
  59. public static string Month(int day = 1, int hour = 1, int minute = 1)
  60. {
  61. return Cron.Monthly(day, hour, minute);
  62. }
  63. /// <summary>
  64. /// 周期性为年的任务
  65. /// </summary>
  66. /// <param name="month">几月开始,默认从一月开始</param>
  67. /// <param name="day">几号开始,默认从一号开始</param>
  68. /// <param name="hour">第几小时开始,默认从1点开始</param>
  69. /// <param name="minute">第几分钟开始,默认从第1分钟开始</param>
  70. /// <returns></returns>
  71. public static string Year(int month = 1, int day = 1, int hour = 1, int minute = 1)
  72. {
  73. return Cron.Yearly(month, day, hour, minute);
  74. }
  75. }
  76. }

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

  1. //MeowvBlogBackgroundJobsModule.cs
  2. ...
  3. public override void OnApplicationInitialization(ApplicationInitializationContext context)
  4. {
  5. var app = context.GetApplicationBuilder();
  6. ...
  7. var service = context.ServiceProvider;
  8. service.UseHangfireTest();
  9. }
  10. ...

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

5

可以看到已经有一个周期性的任务躺在那,每过一分钟都将执行一次,执行完成后如下图,可以很清楚的知道我们的任务当前状态。

6

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

7

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

版权声明:本文为meowv原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/meowv/p/12961014.html