【源码笔记】Nop定时任务
网站需要定时执行不同的任务,比如清理无效的数据、定时发送mail等,Nop的这个定时任务设计比较好,简单的说就是将所有任务相同的属性持久化,具体的执行通过继承接口来实现。
持久化对象:ScheduleTask
ScheduleTask定义了Seconds,Type等属性,分别记录执行周期和任务类型。
public class ScheduleTask:BaseEntity
{
public string Name { get; set; }
/// <summary>
/// Gets or sets the run period (in seconds)
/// </summary>
public int Seconds { get; set; } /// <summary>
/// Gets or sets the type of appropriate ITask class
/// </summary>
public string Type { get; set; } /// <summary>
/// Gets or sets the value indicating whether a task is enabled
/// </summary>
public bool Enabled { get; set; } /// <summary>
/// Gets or sets the value indicating whether a task should be stopped on some error
/// </summary>
public bool StopOnError { get; set; } /// <summary>
/// Gets or sets the machine name (instance) that leased this task. It's used when running in web farm (ensure that a task in run only on one machine). It could be null when not running in web farm.
/// </summary>
public string LeasedByMachineName { get; set; }
/// <summary>
/// Gets or sets the datetime until the task is leased by some machine (instance). It's used when running in web farm (ensure that a task in run only on one machine).
/// </summary>
public DateTime? LeasedUntilTime { get; set; } /// <summary>
/// Gets or sets the datetime when it was started last time
/// </summary>
public DateTime? LastStartTime { get; set; }
/// <summary>
/// Gets or sets the datetime when it was finished last time (no matter failed ir success)
/// </summary>
public DateTime? LastEndTime { get; set; }
/// <summary>
/// Gets or sets the datetime when it was sucessfully finished last time
/// </summary>
public DateTime? LastSuccessTime { get; set; }
}
比如定义一个deleteGuestTask,即2小时删除一次guest。
var deleteGuestTask = new ScheduleTask
{
Name = "Delete Guest Task",
Seconds = ,
Type = "Portal.Services.Users.DeleteGuestTask,Portal.Services",
StopOnError = true,
LeasedByMachineName = "",
Enabled = true,
};
对数据库的基本操作就交给了ScheduleTaskService。ScheduleTaskService继承IScheduleTaskService。
public partial interface IScheduleTaskService
{ void DeleteTask(ScheduleTask task); ScheduleTask GetTaskById(int taskId); ScheduleTask GetTaskByType(string type); IList<ScheduleTask> GetAllTasks(bool showHidden = false); void InsertTask(ScheduleTask task); void UpdateTask(ScheduleTask task);
}
面向接口ITask
ITask接口只有一个方法:
public partial interface ITask
{
/// <summary>
/// Execute task
/// </summary>
void Execute();
}
Nop实现了多个任务

例如DeleteGuestTask:
public class DeleteGuestTask:ITask
{
private readonly IUserService _userService; public DeleteGuestTask(IUserService userService)
{
_userService = userService;
} public void Execute()
{
//60*24 = 1 day
var olderThanMinutes = ;
_userService.DeleteGuestUsers(null, DateTime.Now.AddMinutes(-olderThanMinutes), true);
} }
而ScheduleTask如何和ITask关联的任务就交给了Task类。
public partial class Task
{
/// <summary>
/// Ctor for Task
/// </summary>
private Task()
{
this.Enabled = true;
} /// <summary>
/// Ctor for Task
/// </summary>
/// <param name="task">Task </param>
public Task(ScheduleTask task)
{
this.Type = task.Type;
this.Enabled = task.Enabled;
this.StopOnError = task.StopOnError;
this.Name = task.Name;
} private ITask CreateTask(ILifetimeScope scope)
{
ITask task = null;
if (this.Enabled)
{
var type2 = System.Type.GetType(this.Type);
if (type2 != null)
{
object instance;
if (!EngineContext.Current.ContainerManager.TryResolve(type2, scope, out instance))
{
//not resolved
instance = EngineContext.Current.ContainerManager.ResolveUnregistered(type2, scope);
}
task = instance as ITask;
}
}
return task;
} /// <summary>
/// Executes the task
/// </summary>
/// <param name="scope"></param>
/// <param name="throwException">A value indicating whether exception should be thrown if some error happens</param>
/// <param name="dispose">A value indicating whether all instances hsould be disposed after task run</param>
public void Execute(ILifetimeScope scope=null,bool throwException = false, bool dispose = true)
{
this.IsRunning = true;
//background tasks has an issue with Autofac
//because scope is generated each time it's requested
//that's why we get one single scope here
//this way we can also dispose resources once a task is completed
if (scope == null)
{
scope = EngineContext.Current.ContainerManager.Scope();
}
var scheduleTaskService = EngineContext.Current.ContainerManager.Resolve<IScheduleTaskService>("", scope);
var scheduleTask = scheduleTaskService.GetTaskByType(this.Type); try
{
var task = this.CreateTask(scope);
if (task != null)
{
this.LastStartUtc = DateTime.UtcNow;
if (scheduleTask != null)
{
//update appropriate datetime properties
scheduleTask.LastStartTime = this.LastStartUtc;
scheduleTaskService.UpdateTask(scheduleTask);
} //execute task
task.Execute();
this.LastEndUtc = this.LastSuccessUtc = DateTime.UtcNow;
}
}
catch (Exception exc)
{
this.Enabled = !this.StopOnError;
this.LastEndUtc = DateTime.UtcNow; //log error
if (throwException)
throw;
} if (scheduleTask != null)
{
//update appropriate datetime properties
scheduleTask.LastEndTime = this.LastEndUtc;
scheduleTask.LastSuccessTime = this.LastSuccessUtc;
scheduleTaskService.UpdateTask(scheduleTask);
} //dispose all resources
if (dispose)
{
scope.Dispose();
} this.IsRunning = false;
} /// <summary>
/// A value indicating whether a task is running
/// </summary>
public bool IsRunning { get; private set; } /// <summary>
/// Datetime of the last start
/// </summary>
public DateTime? LastStartUtc { get; private set; } /// <summary>
/// Datetime of the last end
/// </summary>
public DateTime? LastEndUtc { get; private set; } /// <summary>
/// Datetime of the last success
/// </summary>
public DateTime? LastSuccessUtc { get; private set; } /// <summary>
/// A value indicating type of the task
/// </summary>
public string Type { get; private set; } /// <summary>
/// A value indicating whether to stop task on error
/// </summary>
public bool StopOnError { get; private set; } /// <summary>
/// Get the task name
/// </summary>
public string Name { get; private set; } /// <summary>
/// A value indicating whether the task is enabled
/// </summary>
public bool Enabled { get; set; }
}
CreateTask方法用Autofac Resolve(相当于反射)出了对应的任务类型。
private ITask CreateTask(ILifetimeScope scope)
{
ITask task = null;
if (this.Enabled)
{
var type2 = System.Type.GetType(this.Type);
if (type2 != null)
{
object instance;
if (!EngineContext.Current.ContainerManager.TryResolve(type2, scope, out instance))
{
//not resolved
instance = EngineContext.Current.ContainerManager.ResolveUnregistered(type2, scope);
}
task = instance as ITask;
}
}
return task;
}
而Task的Execute方法,调用的是各自任务的Execute。
TaskManager统一管理,TaskThread真正执行
TaskManager统一加载和执行任务,在Global中调用。
protected void Application_Start()
{
//...
TaskManager.Instance.Initialize();
TaskManager.Instance.Start();
}
这里说一下TaskThread,它包含一个Timer和Dictionary<string, Task> _tasks;
public partial class TaskThread : IDisposable
{
private Timer _timer;
private bool _disposed;
private readonly Dictionary<string, Task> _tasks; internal TaskThread()
{
this._tasks = new Dictionary<string, Task>();
this.Seconds = * ;
} private void Run()
{
if (Seconds <= )
return; this.StartedUtc = DateTime.Now;
this.IsRunning = true;
foreach (Task task in this._tasks.Values)
{
task.Execute(Scope,false,false);
}
this.IsRunning = false;
} private void TimerHandler(object state)
{
this._timer.Change(-, -);
this.Run();
if (this.RunOnlyOnce)
{
this.Dispose();
}
else
{
this._timer.Change(this.Interval, this.Interval);
}
} /// <summary>
/// Disposes the instance
/// </summary>
public void Dispose()
{
if ((this._timer != null) && !this._disposed)
{
lock (this)
{
this._timer.Dispose();
this._timer = null;
this._disposed = true;
}
}
} private ILifetimeScope Scope { get; set; } /// <summary>
/// Inits a timer
/// </summary>
public void InitTimer(ILifetimeScope scope)
{
Scope = scope;
if (this._timer == null)
{
this._timer = new Timer(new TimerCallback(this.TimerHandler), null, this.Interval, this.Interval);
}
} /// <summary>
/// Adds a task to the thread
/// </summary>
/// <param name="task">The task to be added</param>
public void AddTask(Task task)
{
if (!this._tasks.ContainsKey(task.Name))
{
this._tasks.Add(task.Name, task);
}
} /// <summary>
/// Gets or sets the interval in seconds at which to run the tasks
/// </summary>
public int Seconds { get; set; } /// <summary>
/// Get or sets a datetime when thread has been started
/// </summary>
public DateTime StartedUtc { get; private set; } /// <summary>
/// Get or sets a value indicating whether thread is running
/// </summary>
public bool IsRunning { get; private set; } /// <summary>
/// Get a list of tasks
/// </summary>
public IList<Task> Tasks
{
get
{
var list = new List<Task>();
foreach (var task in this._tasks.Values)
{
list.Add(task);
}
return new ReadOnlyCollection<Task>(list);
}
} /// <summary>
/// Gets the interval at which to run the tasks
/// </summary>
public int Interval
{
get
{
return this.Seconds * ;
}
} /// <summary>
/// Gets or sets a value indicating whether the thread whould be run only once (per appliction start)
/// </summary>
public bool RunOnlyOnce { get; set; }
}
在Run方法中执行所有包含的任务。这里其实是相同周期的任务。
TaskManager有一个_taskThreads集合,而Initialize方法的主要任务是从数据库加载ScheduleTask.然后将相同周期的任务组装成同一个TaskThread.还区分了只执行一次的任务。
public void Initialize()
{
this._taskThreads.Clear(); var taskService = EngineContext.Current.Resolve<IScheduleTaskService>();
var scheduleTasks = taskService
.GetAllTasks()
.OrderBy(x => x.Seconds)
.ToList(); //group by threads with the same seconds
foreach (var scheduleTaskGrouped in scheduleTasks.GroupBy(x => x.Seconds))
{
//create a thread
var taskThread = new TaskThread
{
Seconds = scheduleTaskGrouped.Key
};
foreach (var scheduleTask in scheduleTaskGrouped)
{
var task = new Task(scheduleTask);
taskThread.AddTask(task);
}
this._taskThreads.Add(taskThread);
} var notRunTasks = scheduleTasks
.Where(x => x.Seconds >= _notRunTasksInterval)
.Where(x => !x.LastStartTime.HasValue || x.LastStartTime.Value.AddSeconds(_notRunTasksInterval) < DateTime.UtcNow)
.ToList();
//create a thread for the tasks which weren't run for a long time
if (notRunTasks.Count > )
{
var taskThread = new TaskThread
{
RunOnlyOnce = true,
Seconds = * //let's run such tasks in 5 minutes after application start
};
foreach (var scheduleTask in notRunTasks)
{
var task = new Task(scheduleTask);
taskThread.AddTask(task);
}
this._taskThreads.Add(taskThread);
}
}
在Star中执行任务。
public void Start()
{
foreach (var taskThread in this._taskThreads)
{
taskThread.InitTimer();
}
}
在后台管理任务的情况,可以立即执行。

public ActionResult RunNow(int id)
{
var scheduleTask = _taskService.GetTaskById(id);
if (scheduleTask == null) return View("NoData");
var task = new Task(scheduleTask);
task.Enabled = true;
task.Execute(null,false,false); return RedirectToAction("Index");
}
个人感觉还是值得借鉴的,我已经用到自己的项目中,可以方便的扩展到自己的其他任务。
Nop源码3.7:http://www.nopcommerce.com/downloads.aspx
【源码笔记】Nop定时任务的更多相关文章
- redis源码笔记(一) —— 从redis的启动到command的分发
		本作品采用知识共享署名 4.0 国际许可协议进行许可.转载联系作者并保留声明头部与原文链接https://luzeshu.com/blog/redis1 本博客同步在http://www.cnblog ... 
- Zepto源码笔记(一)
		最近在研究Zepto的源码,这是第一篇分析,欢迎大家继续关注,第一次写源码笔记,希望大家多指点指点,第一篇文章由于首次分析原因不会有太多干货,希望后面的文章能成为各位大大心目中的干货. Zepto是一 ... 
- AsyncTask源码笔记
		AsyncTask源码笔记 AsyncTask在注释中建议只用来做短时间的异步操作,也就是只有几秒的操作:如果是长时间的操作,建议还是使用java.util.concurrent包中的工具类,例如Ex ... 
- Java Arrays 源码 笔记
		Arrays.java是Java中用来操作数组的类.使用这个工具类可以减少平常很多的工作量.了解其实现,可以避免一些错误的用法. 它提供的操作包括: 排序 sort 查找 binarySearch() ... 
- Tomcat8源码笔记(八)明白Tomcat怎么部署webapps下项目
		以前没想过这么个问题:Tomcat怎么处理webapps下项目,并且我访问浏览器ip: port/项目名/请求路径,以SSM为例,Tomcat怎么就能将请求找到项目呢,项目还是个文件夹类型的? Tom ... 
- Tomcat8源码笔记(七)组件启动Server  Service Engine Host启动
		一.Tomcat启动的入口 Tomcat初始化简单流程前面博客介绍了一遍,组件除了StandardHost都有博客,欢迎大家指文中错误.Tomcat启动类是Bootstrap,而启动容器启动入口位于 ... 
- Tomcat8源码笔记(六)连接器Connector分析
		根据 Tomcat8源码笔记(五)组件Container分析 前文分析,StandardService的初始化重心由 StandardEngine转移到了Connector的初始化,本篇记录下Conn ... 
- Tomcat8源码笔记(五)组件Container分析
		Tomcat8源码笔记(四)Server和Service初始化 介绍过Tomcat中Service的初始化 最先初始化就是Container,而Container初始化过程是咋样的? 说到Contai ... 
- Tomcat8源码笔记(四)Server和Service初始化
		上一章 简单说明下Tomcat各个组件: Server:服务器,Tomcat服务器,一个Tomcat只有一个Server组件; Service:业务层,是Server下最大的子容器,一个Server可 ... 
- Tomcat8源码笔记(三)Catalina加载过程
		之前介绍过 Catalina加载过程是Bootstrap的load调用的 Tomcat8源码笔记(二)Bootstrap启动 按照Catalina的load过程,大致如下: 接下来一步步分析加载过程 ... 
随机推荐
- 前端tip
			background简写 参考地址 http://css.doyoe.com/ 缩写:background:url(test1.jpg) no-repeat scroll 10px 20px/50px ... 
- Sanarus Medical --国外一家研究乳腺癌治疗的科技公司
			Sanarus Medical --国外一家研究乳腺癌治疗的科技公司 http://www.sanarus.com/ 
- qweb
			qweb 是 odoo的模板系统, 在 odoo系统中, 它有不同的用途和实现, 一个是 web client 的 widget 的渲染引擎, 它是通过 javascript实现的,也是 最早引入到 ... 
- 创建nim+安装vioc分区
			一.搭建nim服务器 1.安装须知 安装nim软件包,client随系统默认安装,这里需要安装的是bos.sysmgt.nim.master和bos.sysmgt.nim.spot,即bos.sysm ... 
- Maven入门示例(3):自动部署至外部Tomcat
			Maven入门示例(3):自动部署至外部Tomcat 博客分类: maven 2012原创 Maven入门示例(3):自动部署至外部Tomcat 上一篇,介绍了如何创建Maven项目以及如何在内 ... 
- 1211php面向对象
			首先需要定义数组,$attr = array(直接给元素1,2,3)索引数组 关联数组 $attr = array("one"=>1,2,3) for($i=0;$i< ... 
- eclipse的maven项目报Missing artifact jdk.toos:jdk.toos:jar:1.6错
			很多框架都会依赖jdk中的tools.jar,但是maven仓库中却没有. 如在eclipse+maven编写mapreduce代码,就会报Missing artifact jdk.toos:jdk. ... 
- DWG2SHP DXF2SHP 如何把AutoCAD的DWG,DXF文件转换为Esri ArcGIS的Shape文件
			dwg是AutoCAD创立的一种图纸保存格式,已经成为二维CAD的标准格式,很多其他CAD为了兼容AutoCAD,也直接使用dwg作为默认工作文件. 地图shape文件由ESRI开发,一个ESRI的s ... 
- 【九度OJ】题目1061:成绩排序
			题目描述: 有N个学生的数据,将学生数据按成绩高低排序,如果成绩相同则按姓名字符的字母序排序,如果姓名的字母序也相同则按照学生的年龄排序,并输出N个学生排序后的信息. 输入: 测试数据有多组,每组输入 ... 
- EOS -- 一种灵巧的系统运行跟踪模块
			EOS到底是什么词的缩写,我猜应该是Error of System.最早接触它,是在UT那会.不过那会它是被设计成一个很大的数组,也没有被包含调用函数和行号,又或是时间,只是些计数.编码时,加减一个E ... 
