views:

11583

answers:

14

I have been trying to learn multithreaded programming in C# and am confused about when it is best to use the thread pool vs. create my own threads. One book recommended using the thread pool for small tasks only (whatever that means), but I can't seem to find any real guidelines. What are some considerations you use when making this programming decision?

A: 

Thread pools are great when you have more tasks to process than available threads.

You can add all the tasks to a thread pool and specify the maximum number of threads that can run at a certain time.

Check out this page on MSDN: http://msdn.microsoft.com/en-us/library/3dasc8as(VS.80).aspx

lajos
Ok I guess this ties into my other question. How do you know how many available threads you have at any given time?
Well, it's hard to tell. You'll have to do performance testing. After a point adding more threads will not give you more speed. Find out how many processors are on the machine, that'll be a good starting point. Then go up from there, if processing speed doesn't improve, don't add more threads.
lajos
A: 

Always use a thread pool if you can, work at the highest level of abstraction possible. Thread pools hide creating and destroying threads for you, this is usually a good thing!

JeffFoster
+9  A: 

If you have lots of logical tasks that require constant processing and you want that to be done in parallel use the pool+scheduler.

If you need to make your IO related tasks concurrently such as downloading stuff from remote servers or disk access, but need to do this say once every few minutes, then make your own threads and kill them once you're finished.

Edit: About some considerations, I use thread pools for database access, physics/simulation, AI(games), and for scripted tasks ran on virtual machines that process lots of user defined tasks.

Normally a pool consists of 2 threads per processor (so likely 4 nowadays), however you can set up the amount of threads you want, if you know how many you need.

Edit: The reason to make your own threads is because of context changes, (thats when threads need to swap in and out of the process, along with their memory). Having useless context changes, say when you aren't using your threads, just leaving them sit around as one might say, can easily half the performance of your program (say you have 3 sleeping threads and 2 active threads). Thus if those downloading threads are just waiting they're eating up tons of CPU and cooling down the cache for your real application

Robert Gould
Ok, but can you explain why this is how you approach it? For example, what is the downside of using the thread pool to download from remote servers or do disk IO?
If a thread is waiting on a synchronization object (event, semaphore, mutex, etc) then the thread does not consume CPU.
Brannon
As Brannon said, a common myth is that creation of multiple threads does impact performance. Actually, unused threads consume very few resources. Context switches begins to be a problem only in very high demand servers (in this case, see I/O completion ports for an alternative).
F.D.Castel
Ok, in as you mention in a small application the overhead may be small, but it is there. And in servers its not negligible at all (I work on servers BTW), if you want the real story just ask the folk at Intel and ADM as I've done before.
Robert Gould
Do idle threads impact performance? It depends on how they wait. If well written and waiting on a synchronization object, then they should consume no CPU resources. If waiting in a loop that periodically wakes up to check results, then it is wasting CPU. As always it comes down to good coding.
Bill
+18  A: 

I would suggest you use a thread pool in C# for the same reasons as any other language.

When you want to limit the number of threads running or don't want the overhead of creating and destroying them, use a thread pool.

By small tasks, the book you read means tasks with a short lifetime. If it takes ten seconds to create a thread which only runs for one second, that's one place where you should be using pools (ignore my actual figures, it's the ratio that counts).

Otherwise you spend the bulk of your time creating and destroying threads rather than simply doing the work they're intended to do.

paxdiablo
+11  A: 

Here's a nice summary of the thread pool in .Net: http://blogs.msdn.com/pedram/archive/2007/08/05/dedicated-thread-or-a-threadpool-thread.aspx

The post also has some points on when you should not use the thread pool and start your own thread instead.

Franci Penov
+1 for the link.
SealedSun
-1 for the link. I'm sure it's a good link, but I expect SO to be self-sufficient.
stimpy77
@stimpy77 - that's the wrong expectation then. SO can never be self-sufficient, because it is neither the ultimate authority on all questions, nor all the in-depth information on each topic can (and should) be duplicated in each and every SO answer that touches that topic. (and I don't think you have even enough reputation to downvote every single answer of Jon Skeet alone that has an outbound link, let alone all the answers from all SO users that have outbound links :-))
Franci Penov
Perhaps I was being overly succinct, perhaps I should clarify. I'm not against links. I'm against answers that contain only a link. I don't think that's an answer. Now, if a short summarization of the answer had been posted to summarize how the linked content applies, that would be acceptable. Besides, I came here looking for an answer to the same problem and this answer irritated me because it was yet another link I had to click on to have any idea as to what it might say in reference to the specific problem. Anyway, where does Jon Skeet relate to this? And why should I care?
stimpy77
@stimpy77 - ignore the second half my comment, I was having a bad day and let myself be carried away. apologies for that.The first half still applies though - SO is not the ultimate authority and should not be. The answers are not updated constantly to reflect the constant changes in our field; so anything that is copied here has the chance of being obsolete very soon, sometimes even in a matter of months or weeks. You came to this post two years after it was posted and anything I copied here might have been obsolete by now.
Franci Penov
"You came to this post two years after it was posted and anything I copied here might have been obsolete by now." So might a link. Post a succinct but complete summary when posting a link, you never know if a link goes stale or dead.
stimpy77
+3  A: 

The thread pool is designed to reduce context switching among your threads. Consider a process that has several components running. Each of those components could be creating worker threads. The more threads in your process, the more time is wasted on context switching.

Now, if each of those components were queuing items to the thread pool, you would have a lot less context switching overhead.

The thread pool is designed to maximize the work being done across your CPUs (or CPU cores). That is why, by default, the thread pool spins up multiple threads per processor.

There are some situations where you would not want to use the thread pool. If you are waiting on I/O, or waiting on an event, etc then you tie up that thread pool thread and it can't be used by anyone else. Same idea applies to long running tasks, though what constitutes a long running task is subjective.

Pax Diablo makes a good point as well. Spinning up threads is not free. It takes time and they consume additional memory for their stack space. The thread pool will re-use threads to amortize this cost.

Note: you asked about using a thread pool thread to download data or perform disk I/O. You should not use a thread pool thread for this (for the reasons I outlined above). Instead use asynchronous I/O (aka the BeginXX and EndXX methods). For a FileStream that would be BeginRead and EndRead. For an HttpWebRequest that would be BeginGetResponse and EndGetResponse. They are more complicated to use, but they are the proper way to perform multi-threaded I/O.

Brannon
+1  A: 

Most of the time you can use the pool as you avoid the expensive process of creating the thread.

However in some scenarios you may want to create a thread. For example if you are not the only one using the thread pool and the thread you create is long-lived (to avoid consuming shared resources) or for example if you want to control the stacksize of the thread.

antonio
+1  A: 

One reason to use the thread pool for small tasks only is that there are a limited number of thread pool threads. If one is used for a long time then it stops that thread from being used by other code. If this happens many times then the thread pool can become used up.

Using up the thread pool can have subtle effects - some .NET timers use thread pool threads and will not fire, for example.

Thomas Bratt
A: 

If you have a background task that will live for a long time, like for the entire lifetime of your application, then creating your own thread is a reasonable thing. If you have short jobs that need to be done in a thread, then use thread pooling.

In an application where you are creating many threads, the overhead of creating the threads becomes substantial. Using the thread pool creates the threads once and reuses them, thus avoiding the thread creation overhead.

In an application that I worked on, changing from creating threads to using the thread pool for the short lived threads really helpped the through put of the application.

Bill
Please clarify if you mean "a thread pool" or "the thread pool". These are very different things (at least in the MS CLR).
bzlm
+3  A: 

Beware of the .NET thread pool for operations that may block for any significant, variable or unknown part of their processing, as it is prone to thread starvation. Consider using the .NET parallel extensions, which provide a good number of logical abstractions over threaded operations. They also include a new scheduler, which should be an improvement on ThreadPool. See here

mancaus
We discovered this the hard way! ASP.Net uses the Threadpool is appears and so we couldn't use it as aggressive as we'd like to.
noocyte
A: 

I usually use the Threadpool whenever I need to just do something on another thread and don't really care when it runs or ends. Something like logging or maybe even background downloading a file (though there are better ways to do that async-style). I use my own thread when I need more control. Also what I've found is using a Threadsafe queue (hack your own) to store "command objects" is nice when I have multiple commands that I need to work on in >1 thread. So you'd may split up an Xml file and put each element in a queue and then have multiple threads working on doing some processing on these elements. I wrote such a queue way back in uni (VB.net!) that I've converted to C#. I've included it below for no particular reason (this code might contain some errors).

using System.Collections.Generic;
using System.Threading;

namespace ThreadSafeQueue {
    public class ThreadSafeQueue<T> {
        private Queue<T> _queue;

        public ThreadSafeQueue() {
            _queue = new Queue<T>();
        }

        public void EnqueueSafe(T item) {
            lock ( this ) {
                _queue.Enqueue(item);
                if ( _queue.Count >= 1 )
                    Monitor.Pulse(this);
            }
        }

        public T DequeueSafe() {
            lock ( this ) {
                while ( _queue.Count <= 0 )
                    Monitor.Wait(this);

                return this.DeEnqueueUnblock();

            }
        }

        private T DeEnqueueUnblock() {
            return _queue.Dequeue();
        }
    }
}
noocyte
Some problems with this approach:- Calls to DequeueSafe() will wait until an item is EnqueuedSafe(). Consider using one of the Monitor.Wait() overloads specifying a timeout.- Locking on this is not according to best practices, rather create a readonly object field.- Even though Monitor.Pulse() is lightweight, calling it when the queue contains only 1 item would be more efficient.- DeEnqueueUnblock() should preferrably check the queue.Count > 0. (needed if Monitor.PulseAll or wait timeouts are used)
Craig Nicholson
A: 

Hi,

Don't forget to investigate the Background worker.

I find for a lot of situations, it gives me just what i want without the heavy lifting.

Cheers.

SetiSeeker
A: 

I wanted a thread pool to distribute work across cores with as little latency as possible, and that didn't have to play well with other applications. I found that the .NET thread pool performance wasn't as good as it could be. I knew I wanted one thread per core, so I wrote my own thread pool substitute class. The code is provided as an answer to another StackOverflow question over here.

As to the original question, the thread pool is useful for breaking repetitive computations up into parts that can be executed in parallel (assuming they can be executed in parallel without changing the outcome). Manual thread management is useful for tasks like UI and IO.

cdiggins
+1  A: 

I highly recommend reading the this free e-book: Threading in C# by Joseph Albahari

At least read the "Getting Started" section. The e-book provides a great introduction and includes a wealth of advanced threading information as well.

Knowing whether or not to use the thread pool is just the beginning. Next you will need to determine which method of entering the thread pool best suits your needs:

  • Task Parallel Library (.NET Framework 4.0)
  • ThreadPool.QueueUserWorkItem
  • Asynchronous Delegates
  • BackgroundWorker

This e-book explains these all and advises when to use them vs. create your own thread.

jrupe