Longbow.Tasks

概述

大体分为了Scheduler(调度任务),Storage(持久化),Trigger(触发器),Task(任务)和逻辑模块,大体流程为通过逻辑代码进行实例化相关类,根据ITask来实现独立的业务流程(也可以使用自带业务类),通过配置触发器来设定触发时间,使用持久化来防止程序崩溃导致任务消失,通过调度任务执行任务。

Scheduler

包含了任务调度器的枚举(准备,运行,禁用),调度器类的接口,内部默认实现类和调度执行的实体类

  • IScheduler(任务调度类接口)

    • 包含了调度器状态,下次与您时间,上次执行时间,上次执行结果,调度器创建时间,相关触发器,相关的任务信息
  • DefaultScheduler(内部默认任务调度类)

    • Triggers(相关触发器):SchedulerProcess

    • NextRuntime(下次运行时间):Triggers中可用的最小的运行时间

    • LastRuntime(最近运行时间):Triggers中最后一次运行时间的最大值

      SchedulerProcess?.Triggers.Select(t => t.Trigger.LastRuntime).Max()
    • CreatedTime (创建时间) :DateTimeOffset.Now

  • Start和Stop方法--开始/停止任务调用

    • 该方法是在将调度处理器设置为Running时调用
  • SchedulerProcess(调度执行实体类)

    • 包含了日志委托,任务调度器,调度任务和所有触发器实例

    • Start :调度开始

      public void Start(ITask task, ITrigger trigger){
      Task.Run(() =>
      {
      // 获得任务或 where T:ITask,new()的泛型
      // DefaultTaskMetaData? TaskContext = new DefaultTaskMetaData(task);
      DefaultTaskMetaData? TaskContext = new DefaultTaskMetaData(new T());
      // 将任务赋值给调度器
      _sche.Task = TaskContext.Task;
      _initToken.Cancel(); // Stop 调用
      if (_cancellationTokenSource?.IsCancellationRequested ?? false) return;
      });
      // 设置超时取消
      // 调用Task任务
      // 声明触发器,并注册回调函数开始触发器执行
      InternalStart(trigger);
      }
      // 执行Task并开始触发器
      private void InternalStart(ITrigger trigger)
      {
      //声明一个返回值为Task的,参数为CancellationToken的Func
      var dowork = new Func<CancellationToken, Task>(async token =>
      {
      //没有异常信息
      _sche.Exception = null;
      // 设置任务超时取消令牌
      var taskCancelTokenSource = new CancellationTokenSource(trigger.Timeout);
      try
      {
      // 保证 ITask 的 new() 方法被执行完毕,泛型中有声明
      _initToken.Token.WaitHandle.WaitOne(); var taskToken = CancellationTokenSource.CreateLinkedTokenSource(token, taskCancelTokenSource.Token);
      if (!taskToken.IsCancellationRequested && TaskContext != null)
      {
      //执行异步任务
      await TaskContext.Execute(taskToken.Token).ConfigureAwait(false);
      //异步任务执行完毕,设置为成功
      trigger.LastResult = TriggerResult.Success;
      }
      }
      catch (TaskCanceledException) { }
      catch (Exception ex)
      {
      //有异常,保存异常
      _sche.Exception = ex;
      ex.Log();
      } // 设置 Trigger 状态
      // 取消
      if (token.IsCancellationRequested) trigger.LastResult = TriggerResult.Cancelled;
      // 超时
      if (taskCancelTokenSource.IsCancellationRequested) trigger.LastResult = TriggerResult.Timeout;
      // 错误
      if (_sche.Exception != null) trigger.LastResult = TriggerResult.Error;
      });
      // 声明触发器的维护类
      var triggerProcess = new TriggerProcess(Scheduler.Name, LoggerAction, trigger, Storage, dowork);
      // 保存到触发器数组中
      Triggers.Add(triggerProcess); // 注册触发器状态改变回调方法
      trigger.EnabeldChanged = enabled =>
      {
      // 当触发器状态改变后进行操作
      LoggerAction($"{nameof(TriggerProcess)} Trigger({trigger.GetType().Name}) Enabled({enabled})");
      if (!enabled)
      {
      // 触发器设置为禁用,则持久化的任务设置为停止
      triggerProcess.Stop();
      return;
      }
      // 状态为运行时,则继续运行触发器的维护类(让触发器开始工作)
      if (Status == SchedulerStatus.Running) triggerProcess.Start(_cancellationTokenSource?.Token);
      };
      // 当当前状态为就绪
      if (Status == SchedulerStatus.Ready)
      {
      // 设置为运行
      Status = SchedulerStatus.Running;
      _cancellationTokenSource = new CancellationTokenSource();
      // 运行触发器的维护类
      triggerProcess.Start(_cancellationTokenSource.Token);
      }
      }

Storage

持久化相关类,包含了持久化到文件,序列化触发器,加密等。主要流程为将已初始化的触发器经过加密设置(密钥和向量值)成序列化值,然后将序列化的信息保存到本地文件中的操作

  • FileStorageOptions

    持久化配置信息:是否物理文件持久化(默认为true);保存的文件目录;是否加密(默认true);密钥;向量值

  • IStorage 持久化接口

    /// 从持久化载体加载 ITrigger 触发器
    Task LoadAsync(); /// 将 ITrigger 触发器保存到序列化载体中
    /// <param name="schedulerName">任务调度器名称</param>
    /// <param name="trigger"></param>
    bool Save(string schedulerName, ITrigger trigger); /// 获得 上一次操作异常信息实例
    Exception? Exception { get; } /// 移除指定任务调度
    /// <param name="schedulerNames">要移除调度名称集合</param>
    bool Remove(IEnumerable<string> schedulerNames);
  • FileStorage 持久化到物理文件操作类

    主要方法参数说明

     public virtual Task LoadAsync()
    {
    // 从文件加载
    Exception = null;
    // 物理持久化为true时
    if (Options.Enabled){
    //获得配置中保存的路径,并获得所有*.bin的文件进行遍历
    RetrieveSchedulers().AsParallel().ForAll(fileName =>{
    // 如果存在
    if (File.Exists(fileName)){
    try{
    lock (locker){
    //创建任务名称
    var scheduleName = Path.GetFileNameWithoutExtension(fileName);
    // 反序列化,将string转换为触发器
    var trigger = JsonSerializeExtensions.Deserialize(fileName, Options);
    //获得任务类
    // 返回为null???很奇怪
    var task = CreateTaskByScheduleName(scheduleName);
    if (task != null){
    TaskServicesManager.GetOrAdd(scheduleName, task, trigger);
    }else{
    // 返回为null???很奇怪
    var callback = CreateCallbackByScheduleName(scheduleName);
    if (callback != null){
    TaskServicesManager.GetOrAdd(scheduleName, callback, trigger);
    }
    }
    }
    }
    catch (Exception ex)
    {
    Exception = ex; // load 失败删除文件防止一直 load 出错
    var target = $"{fileName}.err";
    if (File.Exists(target)) File.Delete(target);
    File.Move(fileName, $"{fileName}.err");
    }
    }
    });
    }
    return Task.CompletedTask;
    }
  • FileStorageExtensions

    • ITaskStorageBuilder的扩展类,用于注入物理文件持久化服务到容器
  • JsonSerializeExtensions 二进制序列化操作类

    • 使用TripleDES加密方式

      // 解密
      private static string Decrypte(this string data, FileStorageOptions option)
      {
      using var des = Create(option.Key, option.IV);
      // 创建解密者
      var decryptor = des.CreateDecryptor();
      // 将String转换为byte数组
      var buffer = Convert.FromBase64String(data);
      // 解密数组块
      var result = decryptor.TransformFinalBlock(buffer, 0, buffer.Length);
      return Encoding.UTF8.GetString(result);
      }
      // 加密
      private static string Encrypte(this string data, FileStorageOptions option)
      {
      using var des = Create(option.Key, option.IV);
      //创建加密者
      var encryptor = des.CreateEncryptor();
      //转换
      var buffer = Encoding.UTF8.GetBytes(data);
      // 加密
      var result = encryptor.TransformFinalBlock(buffer, 0, buffer.Length);
      return Convert.ToBase64String(result);
      }
      // 创建TripleDES
      private static TripleDES Create(string key, string iv)
      {
      var des = TripleDES.Create();
      //设置加密模式
      des.Mode = CipherMode.ECB;
      // 填充算法
      des.Padding = PaddingMode.PKCS7;
      // 向量值
      des.IV = Convert.FromBase64String(iv);
      // 加密值
      des.Key = Convert.FromBase64String(key);
      return des;
      }

Trigger

  • ITrigger 触发器接口
  • Cron Cron操作类
  • DefaultTrigger 内部默认的触发器,只执行一次
  • CronTrigger Cron触发器
    • 通过心跳来判断下次执行时间
    • CronExpression.GetNextExecution() 获得下次执行时间
  • RecurringTrigger 重复执行的触发器
    • 通过心跳来判断下次执行时间
    • 等待一个间隔周期,进行触发

Task

  • ITask 任务类接口

逻辑模块

  • TaskServiceCollectionExtensions

    • IServiceCollection注入方法
  • TaskServicesFactory

    • Create
    • StartAsync:调用物理持久化的加载项
    • StopAsync:调用了TaskServicesManager.Shutdown(token)方法
  • TaskServicesManager

    • Init(初始化)

      // 创建任务工厂
      TaskServicesFactory.Create(options, storage ?? new NoneStorage());
      // 加载物理持久化中的触发器
      _ = Factory?.StartAsync();
    • GetOrAdd(新增)

      internal static readonly ConcurrentDictionary<string, Lazy<SchedulerProcess>> _schedulerPool = new();
      // 将任务与触发器添加到调度中 多线程安全
      public static IScheduler GetOrAdd<T>(string schedulerName, ITrigger? trigger = null) where T : ITask, new()
      {
      // 判断任务名称,如果不存在则直接使用任务名称
      if (string.IsNullOrEmpty(schedulerName))
      {
      schedulerName = typeof(T).Name;
      } // 懒加载,因为内部方法中有并发任务,因此需要用懒加载
      return _schedulerPool.GetOrAdd(schedulerName, key => new Lazy<SchedulerProcess>(() =>
      {
      // 创建调度器,并将创建的调度器与任务调度器关联
      var process = GetSchedulerProcess(key); // 绑定任务与触发器
      process.Start<T>(trigger ?? TriggerBuilder.Default.Build());
      return process;
      })).Value.Scheduler;
      }

Longbow.Tasks的更多相关文章

  1. .NET Core 实现后台任务(定时任务)Longbow.Tasks 组件(三)

    原文链接:https://www.cnblogs.com/ysmc/p/16512309.html 在上两篇文章中,简单介绍了怎么使用 IHostedService 与 BackgroundServi ...

  2. .Net多线程编程—System.Threading.Tasks.Parallel

    System.Threading.Tasks.Parallel类提供了Parallel.Invoke,Parallel.For,Parallel.ForEach这三个静态方法. 1 Parallel. ...

  3. spark - tasks is bigger than spark.driver.maxResultSize

    Error ERROR TaskSetManager: Total size of serialized results of 8113 tasks (1131.0 MB) is bigger tha ...

  4. include/linux/tasks.h

    #ifndef _LINUX_TASKS_H#define _LINUX_TASKS_H /* * This is the maximum nr of tasks - change it if you ...

  5. 双核CPU,跑程序会报rcu_sched_state detected stalls on CPUs/tasks 错误

    有一份SDK,之前跑在PPC405EX上没问题。最近换平台,CPU使用了PowerPC的P1020,双核。linux版本也升级到了3.0.48版本。升级之后出现了一个问题:SDK里面的程序跑一段时间之 ...

  6. JavaScript tasks, microtasks, queues and schedules

    最近做的项目中,涉及到了JavaScript中Promise的用法,于是做了一点测试,发现没有想象中的那么简单,水很深,所以找来N先生(我的Mentor),想得到专业的指导.N先生也不尽知,但N先生查 ...

  7. vscode中启动浏览器的tasks.json

    {    // See https://go.microsoft.com/fwlink/?LinkId=733558    // for the documentation about the tas ...

  8. Threading.Tasks 简单的使用

    using Lemon.Common; using System; using System.Collections.Generic; using System.Linq; using System. ...

  9. Effective Java 68 Prefer executors and tasks to threads

    Principle The general mechanism for executing tasks is the executor service. If you think in terms o ...

  10. Tasks.Parallel

    .Net多线程编程-System.Threading.Tasks.Parallel   System.Threading.Tasks.Parallel类提供了Parallel.Invoke,Paral ...

随机推荐

  1. vue-axios删除操作

    <template> <div class="nav"> <input v-model="location" type=" ...

  2. Linux环境下执行脚本重启Weblogic控制台中部署的应用程序

    之前有写过一篇博文介绍切换登录方式的脚本,脚本中存在一个缺点:仍需手动去Weblogic控制台重启应用程序:本文即介绍如何在脚本中更新Weblogic控制台中部署的应用程序. 一.配置Weblogic ...

  3. OpenMP 教程(一) 深入人剖析 OpenMP reduction 子句

    OpenMP 教程(一) 深入人剖析 OpenMP reduction 子句 前言 在前面的教程OpenMP入门当中我们简要介绍了 OpenMP 的一些基础的使用方法,在本篇文章当中我们将从一些基础的 ...

  4. 记录一次新节点加入K8S集群

    新节点初始化 安装docker kubelet kubeadm(指定版本) #先查看当前集群docker版本 [root@lecode-k8s-master manifests]# docker ve ...

  5. vulnhub靶场之DEATHNOTE: 1

    准备: 攻击机:虚拟机kali.本机win10. 靶机:DEATHNOTE: 1,网段地址我这里设置的桥接,所以与本机电脑在同一网段,下载地址:https://download.vulnhub.com ...

  6. 4.mysql-进阶

    1.事务 将多个操作步骤变成一个事务,任何一个步骤失败,则回滚到事务的所有步骤之前状态,大白话:要成功都成功:要失败都失败. 如转账操作,A扣钱.B收钱,必须两个步骤都成功,才认为转账成功 innod ...

  7. lvm+xfs的扩缩容

    ext4文件系统可以经行扩缩容操作,但xfs的文件系统只能扩容,无法缩容 所以如果需要进行xfs的缩容,可以先使用xfsdump备份文件系统,然后对逻辑卷(/分区)进行缩容操作(此时原xfs文件系统会 ...

  8. Android网络请求(3) 网络请求框架OkHttp

    Android网络请求(3) 网络请求框架OkHttp 本节我们来讲解OkHtpp网络请求框架 什么是网络请求框架 在我的理解中,网络请求框架是为了方便我们更加便捷规范的进行网络请求所建的类,我们通过 ...

  9. Google Chrome(谷歌浏览器)安装使用

    谷歌浏览器官网https://www.google.cn/chrome/ Chrome是由Google开发的一款简单便捷的网页浏览工具.谷歌浏览器(Google Chrome)可以提帮助你快速.安全的 ...

  10. 1.1 大数据简介-hadoop-最全最完整的保姆级的java大数据学习资料

    目录 1 hadoop-最全最完整的保姆级的java大数据学习资料 1.1 大数据简介 1.1.1 大数据的定义 1.1.2 大数据的特点 1.1.3 大数据的应用场景 1.1.4 大数据的发展趋势及 ...