使用 Quartz.NET 3.0.3 和 Simple Injector How To 进行构造函数注入

Dar*_*ren 3 c# windows-services dependency-injection quartz.net simple-injector

我正在尝试在 Windows 服务中使用 Quartz.Net v3.0.3 和 Simple Injector。

我有一个工作类,我想在其中注入一些依赖项,例如我的记录器。

public class JobWorker :  IJob
{
    private ILogger _logger;

    public JobWorker(ILogger logger)
    {
        _logger = logger;
    }

    public Task Execute(IJobExecutionContext context)
    {
        return Task.Run(() =>_logger.Log("Do Work"));
    }
}
Run Code Online (Sandbox Code Playgroud)

我尝试Container.Register<IJob, JobWorker>();在我的 DI 层上注册,但这没有帮助。

如果我删除注入的依赖项并简单地使用默认的无参数构造函数,则作业会正确触发。

根据 Steven 下面的帖子,建议是创建一个工厂,但是在新框架的上下文中提供的答案已经过时,我完全不知道如何将依赖项注入作业。

使用 Quartz.NET 和 Simple Injector 进行构造函数注入

Lin*_*res 5

@Rabban 提供的链接仍然有效,使用IServiceProvider是一个不错的设计选择,但您可以使用任何您想要的具体容器。

这是我的 2c 基于 Rabban 使用Quartz 3.0.4and的回答SimpleInjector 4.2.1

using NLog;
using Quartz;
using Quartz.Spi;
using System;

namespace My.Dear.App.Infrastructure
{

    public class SomeJobFactory : IJobFactory
    {
        private static ILogger logger = LogManager.GetCurrentClassLogger();
        private readonly IServiceProvider serviceProvider;

        public DexJobFactory(IServiceProvider serviceProvider)
        {
            this.serviceProvider = serviceProvider;
        }

        public IJob NewJob(TriggerFiredBundle bundle, IScheduler scheduler)
        {
            try
            {
                IJobDetail jobDetail = bundle.JobDetail;
                Type jobType = jobDetail.JobType;
                logger.Debug($"Producing instance of Job '{jobDetail.Key}', class={jobType.FullName}");

                return serviceProvider.GetService(jobType) as IJob;
            }
            catch (Exception ex)
            {
                logger.Error(ex, Constants.ErrorAt, nameof(IJobFactory.NewJob));
                throw new SchedulerException($"Problem instantiating class '{bundle.JobDetail.JobType.FullName}'", ex);
            }
        }

        public void ReturnJob(IJob job)
        {
            var disposable = job as IDisposable;
            disposable?.Dispose();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

对我来说就像一种魅力。

如何获取实例?

public static async Task RegisterQuartz(Container container)
{
    ISchedulerFactory schedulerFactory = new StdSchedulerFactory();
    IScheduler scheduler = await schedulerFactory.GetScheduler();
    IJobFactory jobFactory = new SomeJobFactory(container);
    scheduler.JobFactory = jobFactory;

    container.RegisterInstance(schedulerFactory);
    container.RegisterInstance(jobFactory);
    container.RegisterInstance(scheduler);
    container.Register<IDearJob, DearJob>();
}
Run Code Online (Sandbox Code Playgroud)

哦,别忘了注册你的工作。否则它可能无法工作。

我建议为每个作业创建一个接口,而不是Quartz IJob为此使用。

public interface IDearJob : IJob { }

public interface DearJob : IDearJob 
{
    private readonly ISomeService service;

    public DearJob(ISomeService service)
    {
        this.service = service;
    }

    public async Task Execute(IJobExecutionContext context) 
    {
        // retrieve context if you need
        await this.service.DoSomethingAsync(/*params*/);
    }
}
Run Code Online (Sandbox Code Playgroud)

现在您可以在 Execute 上使用断点。干杯。

编辑

PS:史蒂文的回答非常好,我认为你可以用它来更新你的上下文。现在是认真的,干杯。