c# TaskScheduler

这里记录下 TaskScheduler 的简单用法。

使用场景:

使用 Task 的时候,大家知道用 TaskFactory.StartNew 可以用来创建一个 Task 。这里如果创建了 3 个,那么这3个 Task 就各自放飞直接运行了。

csharp 复制代码
class Program
{
	private static TaskFactory _taskFactory;
	static void Main(string[] args)
	{
		_taskFactory = new TaskFactory();
 
		_taskFactory.StartNew(Func1);
		_taskFactory.StartNew(Func2);
		_taskFactory.StartNew(Func3);
 
		Console.ReadLine();
	}
 
	static void Func1()
	{
		Thread.Sleep(3000);
		Console.WriteLine("Func1");
	}
	static void Func2()
	{
		Thread.Sleep(2000);
		Console.WriteLine("Func2");
	}
	static void Func3()
	{
		Thread.Sleep(1000);
		Console.WriteLine("Func3");
	}
}

结果:

Func3

Func2

Func1

一般情况下没什么大问题,但如果这3个中的每个Task都非常耗CPU或者内存,而计算机又需要预留资源去干别的事情,这就要让3个Task不能同时执行。又或者确实要让某些Task先做,有些后做。这就需要我们自己能够决定Task执行顺序。

怎样达到这样的效果?

答案就是:TaskScheduler,它可以让已经创建好的 Task 去按照特殊的顺序来执行。

就拿上面的场景来举例:

为了节约系统资源,我要让这3个Task单独执行,有一个在执行,其它俩就不能执行。最简单的就是串行执行,这样只需要写一个类继承TaskScheduler:

csharp 复制代码
public class MyTaskScheduler : TaskScheduler, IDisposable
{
	private static readonly object _mutex = new object();
	public readonly List<Task> _currentTasks = new List<Task>();
	private readonly ManualResetEvent[] _schedulerEvents = new ManualResetEvent[2];
	public override int MaximumConcurrencyLevel => 1;
 
	public MyTaskScheduler()
	{
		_schedulerEvents[0] = new ManualResetEvent(false);
		_schedulerEvents[1] = new ManualResetEvent(false);
 
		var executionThread = new Thread(ExecutionThread) { Name = "MyThread" };
		executionThread.SetApartmentState(ApartmentState.MTA);
		executionThread.IsBackground = true;
		executionThread.Priority = ThreadPriority.Normal;
		executionThread.Start(null);
	}
 
	public void Dispose()
	{
		_schedulerEvents[1].Set();
	}
 
	private void ExecutionThread(object args)
	{
		try
		{
 
			while (true)
			{
				if (!WaitIfEmpty())
				{
					break;
				}
 
				Task task;
				lock (_mutex)
				{
					task = _currentTasks[0];
				}
 
				try
				{
					TryExecuteTask(task);
				}
				catch (Exception)
				{
					throw;
				}
				finally
				{
					TryDequeue(task);
				}
			}
		}
		finally
		{
			//todo
		}
	}
	private bool WaitIfEmpty()
	{
		lock (_mutex)
		{
			if (_currentTasks.Count == 0)
			{
				//pause task
				_schedulerEvents[0].Reset();
			}
		}
 
		//wait any signal.
		int id = WaitHandle.WaitAny(_schedulerEvents);
 
		// id is item index of _schedulerEvents
		return id == 0;
	}
 
	protected override IEnumerable<Task>? GetScheduledTasks()
	{
		lock (_mutex)
		{
			return _currentTasks.ToArray();
		}
	}
	protected override void QueueTask(Task task)
	{
		lock (_mutex)
		{
			_currentTasks.Add(task);
 
			if (_currentTasks.Count > 0)
			{
				// Start task
				_schedulerEvents[0].Set();
			}
		}
	}
	protected sealed override bool TryDequeue(Task task)
	{
		lock (_mutex)
		{
			var res = _currentTasks.Remove(task);
			return res;
		}
	}
	protected override bool TryExecuteTaskInline(Task task, bool taskWasPreviouslyQueued)
	{
		return false;
	}
}

运行结果:

Func1

Func2

Func3

在MyTaskScheduler 中,我建了一个线程 executionThread,线程方法是一个while循环,一直在监听有没有新的Task过来,有多个Task过来,就按Task创建顺序执行,没有的话 while 就暂停在 waitany 处。

这样的场景,配合 ManualResetEvent 和Task的CancellationTokenSource 非常适合做 带有暂停和取消功能的任务列表。

相关推荐
CallZhang21011 小时前
Vision Master的C#脚本与opencv联合编程
opencv·计算机视觉·c#·视觉检测
AI视觉网奇11 小时前
kafka 冲突解决 kafka安装
c#·linq
hqwest12 小时前
C#WPF实战出真汁07--【系统设置】--菜品类型设置
开发语言·c#·wpf·grid设计·stackpanel布局
萘柰奈12 小时前
Unity进阶--C#补充知识点--【Unity跨平台的原理】Mono与IL2CPP
unity·c#·游戏引擎
程序设计实验室13 小时前
StarBlog v1.3.0 新版本,一大波更新以及迁移服务器部署
c#·aspnetcore·starblog番外
淡海水13 小时前
【原理】Struct 和 Class 辨析
开发语言·c++·c#·struct·class
淡海水13 小时前
【原理】Unity GC 对比 C# GC
unity·c#·gc·垃圾回收
张人玉14 小时前
C#读取文件, IO 类属性及使用示例
microsoft·c#
咕白m62519 小时前
通过 C# 高效提取 PDF 文本的完整指南
后端·c#
hqwest1 天前
C#WPF实战出真汁08--【消费开单】--餐桌面板展示
c#·wpf·ui设计·wpf界面设计