如何从自定义TaskScheduler回退到Default TaskScheduler

时间:2014-03-22 21:31:04

标签: c# .net task-parallel-library

让我们说我想创建一个自定义的TaskScheduler,但在其中,如果某些条件不适用,则回退到默认值。我该怎么做?

E.g。

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? */);
}

1 个答案:

答案 0 :(得分:2)

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

但是,您不能,因为TPL中TaskScheduler的{​​{3}}是privateinternal

如果使用TaskScheduler.QueueTask,可能诱人只需将呼叫转发至all non-abstract implementations

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

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

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

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

因此,QueueTaskGetScheduledTasksTaskScheduler的其他一些方法被声明为protected是一个很好的理由。如果您要实施自定义任务计划程序,则必须提供完整的实施。