c# – 如何从自定义TaskScheduler回退到Default TaskScheduler

假设我想创建一个自定义TaskScheduler,但在其中,如果某些条件不适用,则回退到默认值.我怎么做?

例如.

protected override void QueueTask(Task task) {
   if (/* some condition */) {
       /* schedule task manually, does not matter */
   }

   /* queue task on default -- how? */
}

protected override IEnumerable<Task> GetScheduledTasks() {
    return Enumerable.Concat(/* my tasks */, /* tasks from default -- how? */);
}

最佳答案 我不认为你可以(或应该)这样做,即使你使用反射黑客.理论上,您需要从现有的非抽象任务调度程序派生自定义TaskScheduler,例如,
ThreadPoolTaskScheduler.

但是,你不能因为TPL中的TaskScheduler的all non-abstract implementations是私有的或内部的.

对于TaskScheduler.QueueTask,可能很容易将调用转发到Task.Start

protected override void QueueTask(Task task) {
   if (/* some condition */) {
       /* schedule task manually, does not matter */
   }

   /* queue task on default -- how? */
   task.Start(TaskScheduler.Default);
}

但是,你也不能这样做. implementation of Task.Start在内部调用TaskScheduler.QueueTask,这将导致无限递归.

此外,Task对象在Task.Start中存储对任务调度程序的引用.因此,如果您使用反射来调用ThreadPoolTask​​Scheduler.QueueTask,它将破坏任务与其任务调度程序之间的连接.

因此,有一个很好的理由将QueueTask,GetScheduledTasks和TaskScheduler的其他一些方法声明为protected.如果您要实现自定义任务计划程序,则必须提供完整的实现.

点赞