is returning an empty static task in TPL a bad pra

2020-06-08 14:50发布

问题:

There are cases that I would want to run a task conditionally. I use some sort of extension method like this:

public static class MyTaskExtension{
  private static Task theEmptyTask = Task.Factory.StartNew(() => {}); //This is the question

  public static Task ContinueWith(this Task task, Task continuationTask, Func<bool> condition)
  {
    if condition(){
       ... do the work
    }
    return theEmptyTask;
  }
}

My expectation is that theEmptyTask would be already completed, so basically if I don't want to do anything I just return this task, instead of null or a new empty task.

I have a feeling that there should be some glitch with this approach. Can anyone see that?

回答1:

It's perfectly acceptable to return an already completed task in some contexts. It's not something that is done particularly often, but it is done.

There's also nothing wrong at all with just using a single static completed task. There is no need to have a whole bunch of different tasks that are all identical, given that once they're completed, and if they have no result, there's nothing wrong with reusing them.

Note that if you want to return an already completed task you can use Task.FromResult to generate one with less overhead than what you're doing now, as you won't be creating an empty method, scheduling it, waiting for it to be started, and then have it finish right away. Just returning Task.FromResult(false) will give you an already completed task.

If you are using .NET 4.0 you can create your own FromResult easily enough:

public static Task FromResult<T>(T result)
{
    var tcs = new TaskCompletionSource<T>();
    tcs.SetResult(result);
    return tcs.Task;
}


回答2:

As long as you hand back a task that is in the completed state (use TaskCompletionSource to do this), I can't think of any problem with this since there aren't really any setters on the Task class that would allow the client to muck with your static empty task. They could call Dispose() on your task, but I don't think that would cause any harm (i.e. I don't think it would affect the ability to inspect the properties of the Task (haven't tried it out--something worth testing out)).