Asp .Net Core 系列:Asp .Net Core 集成 Hangfire+MySQL

在 .NET 或 .NET Core 应用中,若需在不依赖 Windows 服务、独立进程 的前提下实现后台处理,Hangfire 是最成熟、简单的方案之一 ------ 它可直接嵌入现有应用(如 ASP.NET Core Web 应用),无需额外部署,同时提供任务持久化、监控和重试能力。以下从 核心优势、快速集成步骤、关键用法 三方面,梳理如何用 Hangfire 实现后台处理:

一、为什么选择 Hangfire(核心优势)

  1. 零额外部署:可直接嵌入 Web 应用、控制台应用,无需单独部署 Windows 服务或计划任务;
  2. 任务持久化:支持 SQL Server、MySQL、Redis 等存储,应用重启后未执行的任务不会丢失;
  3. 可视化监控:自带 Dashboard 控制台,可实时查看任务状态(成功 / 失败 / 排队)、执行日志、重试记录;
  4. 多任务类型支持:覆盖常见后台场景(一次性任务、定时任务、周期性任务);
  5. 高可靠性:自动重试失败任务,支持任务优先级(队列区分),避免单点故障。
bash 复制代码
Install-Package Hangfire.AspNetCore
Install-Package Hangfire.MySqlStorage
cs 复制代码
    #region Hangfire与MySQL
    string connString = configuration.GetConnectionString("Hangfire")
                       ?? throw new ArgumentException("未找到Hangfire连接字符串");
    context.Services.AddHangfire(config =>
    {
        config.SetDataCompatibilityLevel(CompatibilityLevel.Version_180)
              .UseSimpleAssemblyNameTypeSerializer()
              .UseRecommendedSerializerSettings();
        config.UseStorage(
                  new MySqlStorage(connString, new MySqlStorageOptions()));
    });
    context.Services.AddHangfireServer();
    context.Services.AddTransient<IBackgroundJobClient, BackgroundJobClient>();
    #endregion

    #region 定义Queues
    context.Services.AddHangfireServer(options =>
    {
        options.Queues = new[] { "maintenance-plan-overtime" };
        options.WorkerCount = 1;
    });
    #endregion
}

public override async Task OnApplicationInitializationAsync(
ApplicationInitializationContext context)
{
    await context.AddBackgroundWorkerAsync<LimitRecordWorker>();
}

public override Task OnPostApplicationInitializationAsync(ApplicationInitializationContext context)
{
    const string NAME = "HangfirePeriodicBackgroundWorkerAdapter<BackgroundJobWorker>.DoWorkAsync";
    RecurringJob.RemoveIfExists(NAME);
    return base.OnPostApplicationInitializationAsync(context);
}

如果你想使用 Hangfire Dashboard 来查看和管理后台任务,你需要在 Configure 方法中添加相应的中间件:

cs 复制代码
app.UseHangfireDashboard();

定时任务:

cs 复制代码
public class LimitRecordWorker : HangfireBackgroundWorkerBase
{
    private readonly IAbpDistributedLock _distributedLock;
    private readonly IUnitOfWorkManager _unitOfWorkManager;
    private readonly IBackgroundJobManager _backgroundJobManager;

    public LimitRecordWorker(IAbpDistributedLock distributedLock,
        IUnitOfWorkManager unitOfWorkManager,
        IConfiguration configuration,
        IBackgroundJobManager backgroundJobManager)
    {
        _distributedLock = distributedLock;
        _unitOfWorkManager = unitOfWorkManager;
        RecurringJobId = nameof(LimitRecordWorker);
        _backgroundJobManager = backgroundJobManager;
        CronExpression = configuration["Crons:LimitRecordCron"];
    }

    public override async Task DoWorkAsync(CancellationToken cancellationToken = default)
    {
        await using (var handle = await _distributedLock
            .TryAcquireAsync("LimitRecordDistributedLock", TimeSpan.FromSeconds(10)))
        {
            if (handle != null)
            {
                using (var uow = LazyServiceProvider.LazyGetRequiredService<IUnitOfWorkManager>().Begin())
                {
                    Log.Information("LimitRecordWorker" + DateTime.Now);

                    for (int i = 0; i <= 5; i++)
                    {
                        MaintenancePlanOverTimeArgs workProcedureLeaderArgs = new MaintenancePlanOverTimeArgs();
                        workProcedureLeaderArgs.Serial = i;
                        await _backgroundJobManager.EnqueueAsync(workProcedureLeaderArgs, delay: TimeSpan.FromHours(workProcedureLeaderArgs.OverTime % 24));
                    }
                    await uow.CompleteAsync();
                }
            }
        }
    }
}

队列服务:

cs 复制代码
    [Queue("maintenance-plan-overtime")]
    public class MaintenancePlanOverTimeJob : AsyncBackgroundJob<MaintenancePlanOverTimeArgs>, ITransientDependency
    {
        public MaintenancePlanOverTimeJob()
        {
        }

        public override async Task ExecuteAsync(MaintenancePlanOverTimeArgs args)
        {
            Log.Information("MaintenancePlanOverTimeJob:" + args.Serial + "--" + DateTime.Now);
        }
    }

    public class MaintenancePlanOverTimeArgs
    {
        public int Serial { get; set; }
        public List<Guid> EmployeeIds { get; set; }
        public Guid DepartmentId { get; set; }
        public Guid? WorkProcedureId { get; set; }
        public Guid? DeviceDetailTypeId { get; set; }
        public double OverTime { get; set; }
    }

效果:

相关推荐
dfdfadffa10 小时前
如何创建仅在首次订阅时执行一次计算的 RxJS 懒加载 Observable
jvm·数据库·python
Irene199110 小时前
Oracle 中:为什么 from 子查询后面需要一个别名
数据库·oracle
m0_6245785910 小时前
SQL分组后如何计算移动平均值_利用窗口函数AVG配合ROWS
jvm·数据库·python
2401_8242226910 小时前
如何修复待办事项列表无法添加任务的 JavaScript 错误
jvm·数据库·python
地球资源数据云10 小时前
1900-2023年中国物种分布点位矢量数据集
大数据·数据结构·数据库·数据仓库·人工智能
tang&11 小时前
【MySQL】索引创建与B+树原理:MySQL性能优化的核心一课
b树·mysql·性能优化
sitellla11 小时前
MySQL 入门:最流行的开源关系型数据库介绍
数据库·mysql·其他·开源
精益数智工坊11 小时前
拆解制造业仓库物料管理流程:如何通过标准化仓库物料管理流程解决账实不符难题
大数据·前端·数据库·人工智能·精益工程
nbwenren11 小时前
办公AI实测:Gemini3、GPT-4o、Claude3.5谁更强?
服务器·数据库·php
2401_8242226911 小时前
如何卸载并重装Oracle Grid_Deinstall脚本与ASM磁盘清理
jvm·数据库·python