如何在作业执行期间将数据保存到 JobDataMap 并在之后访问它?

How to save data to JobDataMap during job execution and access it after?

我目前正在围绕 Quartz.net 开发某种简化的包装器,以便能够管理所有已注册和 运行 的后台作业,显示有关作业执行进度、百分比的其他信息,return 关于当前工作状态等的某种消息。例如,如果我当前的工作是将数据发布到社交媒体,我想查看当前发布到哪个社交媒体的数据,或者如果工作正在发送电子邮件我想实时查看当前生成了多少封电子邮件,已经发送了多少封,还剩多少封等等。为此,我将数据保存到作业中的IJobExecutionContext (context.JobDetail.JobDataMap) Execute 方法。然后使用我的管理器获取所有当前正在执行的作业:

public List<IJobExecutionContext> RunningJobs
{
     get { return Scheduler.GetCurrentlyExecutingJobs().ToList(); }
}

并显示从每个 IJobExecutionContextJobDataMap 中获取的详细信息。这有点工作正常。但问题是,例如,如果作业执行失败,我想在 JobDataMap 中记录(保存)尽可能多的数据,包括异常对象本身。但是,问题是,当作业执行完成时,上下文对象被释放。我丢失了所有这些数据。因此,当我尝试获取当前计划作业的所有 JobDetail 时:

public List<IJobDetail> AllJobs
{
     get
     {
         var result = new List<IJobDetail>();
         var jobGroups = Scheduler.GetJobGroupNames();

         foreach(string group in jobGroups)
         {
            var groupMatcher = GroupMatcher<JobKey>.GroupContains(group);
            var jobKeys = Scheduler.GetJobKeys(groupMatcher);

            foreach(var jobKey in jobKeys)
            {
                var detail = Scheduler.GetJobDetail(jobKey);
                result.Add(detail);
            }
         }

         return result;
     }
}

我看不到我在作业执行期间保存到 detail.JobDataMap 的数据。 但是我仍然可以看到作业 initialized/scheduled 时在实际执行之前保存的数据。作为一个选项,我可以在执行期间捕获异常并记录它(我实际上正在做),但我想将此作业标记为失败并在我的作业详细信息视图中显示失败的原因。它使用 List<IJobDetail> AllJobs (在上面实现)从我的经理(就像我的工作的某种存储库)获取数据。

所以我必须使用某种额外的作业详细信息存储来在执行期间保存此数据,然后获取所有计划的作业并通过作业 ID 或其他方式映射此数据。这不是我想要的解决方案。或者实际上有一种方法可以在执行时将数据保存到 JobDataMap 并在执行后访问此数据?

经过一些研究,我发现 Quartz 中有两种类型的作业:有状态和无状态(无状态作业)。

Stateful job instances follow slightly different rules from regular Job instances. The key difference is that their associated JobDataMap is re-persisted after every execution of the job, thus preserving state for the next execution. The other difference is that stateful jobs are not allowed to execute concurrently, which means new triggers that occur before the completion of the execute(xx) method will be delayed.

因此,为了实现我的目标,我只需要使用 PersistJobDataAfterExecutionDisallowConcurrentExecution 属性来装饰我的工作 class,这将使我的工作有状态,并且它会在执行后保留它的 JobDataMap。