Does using Tasks (TPL) library make an application multithreaded?

Tasks can be used to represent operations taking place on multiple threads, but they don’t have to. One can write complex TPL applications that only ever execute in a single thread. When you have a task that, for example, represents a network request for some data, that task is not going to create additional threads to accomplish that goal. Such a program is (hopefully) asynchronous, but not necessarily mutlithreaded.

Parallelism is doing more than one thing at the same time. This may or may not be the result of multiple threads.

Let’s go with an analogy here.


Here is how Bob cooks dinner:

  1. He fills a pot of water, and boils it.
  2. He then puts pasta in the water.
  3. He drains the pasta when its done.
  4. He prepares the ingredients for his sauce.
  5. He puts all of the ingredients for his sauce in a saucepan.
  6. He cooks his sauce.
  7. He puts his sauce on his pasta.
  8. He eats dinner.

Bob has cooked entirely synchronously with no multithreading, asynchrony, or parallelism when cooking his dinner.


Here is how Jane cooks dinner:

  1. She fills a pot of water and starts boiling it.
  2. She prepares the ingredients for her sauce.
  3. She puts the pasta in the boiling water.
  4. She puts the ingredients in the saucepan.
  5. She drains her pasta.
  6. She puts the sauce on her pasta.
  7. She eats her dinner.

Jane leveraged asynchronous cooking (without any multithreading) to achieve parallelism when cooking her dinner.


Here is how Servy cooks dinner:

  1. He tells Bob to boil a pot of water, put in the pasta when ready, and serve the pasta.
  2. He tells Jane to prepare the ingredients for the sauce, cook it, and then serve it over the pasta when done.
  3. He waits for Bob and Jane to finish.
  4. He eats his dinner.

Servy leveraged multiple threads (workers) who each individually did their work synchronously, but who worked asynchronously with respect to each other to achieve parallelism.

Of course, this becomes all the more interesting if we consider, for example, whether our stove has two burners or just one. If our stove has two burners then our two threads, Bob and Jane, are both able to do their work without getting in each others way, much. They might bump shoulders a bit, or each try to grab something from the same cabinet every now and then, so they’ll each be slowed down a bit, but not much. If they each need to share a single stove burner though then they won’t actually be able to get much done at all whenever the other person is doing work. In that case, the work won’t actually get done any faster than just having one person doing the cooking entirely synchronously, like Bob does when he’s on his own. In this case we are cooking with multiple threads, but our cooking isn’t parallelized. Not all multithreaded work is actually parallel work. This is what happens when you are running multiple threads on a machine with one CPU. You don’t actually get work done any faster than just using one thread, because each thread is just taking turns doing work. (That doesn’t mean multithreaded programs are pointless on one cores CPUs, they’re not, it’s just that the reason for using them isn’t to improve speed.)


We can even consider how these cooks would do their work using the Task Parallel Library, to see what uses of the TPL correspond to each of these types of cooks:

So first we have bob, just writing normal non-TPL code and doing everything synchronously:

public class Bob : ICook
{
    public IMeal Cook()
    {
        Pasta pasta = PastaCookingOperations.MakePasta();
        Sauce sauce = PastaCookingOperations.MakeSauce();
        return PastaCookingOperations.Combine(pasta, sauce);
    }
}

Then we have Jane, who starts two different asynchronous operations, then waits for both of them after starting each of them to compute her result.

public class Jane : ICook
{
    public IMeal Cook()
    {
        Task<Pasta> pastaTask = PastaCookingOperations.MakePastaAsync();
        Task<Sauce> sauceTask = PastaCookingOperations.MakeSauceAsync();
        return PastaCookingOperations.Combine(pastaTask.Result, sauceTask.Result);
    }
}

As a reminder here, Jane is using the TPL, and she’s doing much of her work in parallel, but she’s only using a single thread to do her work.

Then we have Servy, who uses Task.Run to create a task that represents doing work in another thread. He starts two different workers, has them each both synchronously do some work, and then waits for both workers to finish.

public class Servy : ICook
{
    public IMeal Cook()
    {
        var bobsWork = Task.Run(() => PastaCookingOperations.MakePasta());
        var janesWork = Task.Run(() => PastaCookingOperations.MakeSauce());
        return PastaCookingOperations.Combine(bobsWork.Result, janesWork.Result);
    }
}

Leave a Comment