Blue Theme Orange Theme Green Theme Red Theme
 
Home | Forums | Videos | Photos | Downloads | Blogs | Interviews | Jobs | Beginners | Training
 | Consulting  
Submit an Article Submit a Blog 
 Login Close
User Id:
Password:
 
Forgot Password
Forgot Username
Why Register
 Jump to
Skip Navigation Links
TechnologyExpand Technology
WebsiteExpand Website
 Resources  
Close
 Our Network  
Close
Search :       Advanced Search »
Home » Threading » Introduction to Multithreading in C#

Introduction to Multithreading in C#

This article is a complete introdution to threading. It explains what is a thread and why it is used in programming. It also talks about common problems with threading, and how to begin writing thread safe code.

Author Rank:
Total page views :  192531
Total downloads :  2225
   Print Read/Post comments Post a comment  Similar Articles  
   Email to a friend  Bookmark  Author's other articles  
Download Files:
multithreading.ZIP
 
Become a Sponsor

In working in New York and talking to programmers all over Wall Street, I've noticed a common thread of knowledge expected in most real time programming applications.  That knowledge is known as multithreading.  As I have migrated around the programming world, and performed interviews on potential programming candidates, it never ceases to amaze me how little is known about multithreading or why or how threading is applied.  In a series of excellent articles written by Vance Morrison, MSDN has tried to address this problem:  (See the August Issue of MSDN, What every Developer Must Know about Multithreaded Apps,  and the October issue Understand the Impact of Low-Lock Techniques in Multithreaded Apps. Also there are many articles here on C# Corner on threading such as Understanding Threading in the .NET Framework by Chandrakant Parmar or the four part series by Manisha Mehta: Multithreading Part 1: Multithreading and Multitasking,    Multithreading Part 2: Understanding the System.Threading.Thread ClassMultithreading Part 3: Thread Synchronization, and Multithreading Part 4: The ThreadPool, Timer Classes and Asynchronous Programming Discussed.

In this article I will attempt to give an introductory discussion on threading, why it is used, and how you use it in .NET.  I hope to once and for all unveil the mystery behind multithreading, and in explaining it, help avert potential threading disasters in your code.

What is a thread?

Every application runs with at least one thread.  So what is a thread?  A thread is nothing more than a process.  My guess is that the word thread comes from the Greek mythology of supernatural Muses weaving threads on a loom, where each thread represents a path in time of someone's life.  If you mess with that thread, then you disturb the fabric of life or change the process of life.  On the computer, a thread is a process moving through time.  The process performs sets of sequential steps, each step executing a line of code.  Because the steps are sequential, each step takes a given amount of time.  The time it takes to complete a series of steps is the sum of the time it takes to perform each programming step.

What are multithreaded applications?

For a long time, most programming applications (except for embedded system programs) were single-threaded.  That means there was only one thread in the entire application.  You could never do computation A until completing computation B.  A program starts at step 1 and continues sequentially (step 2, step 3, step 4) until it hits the final step (call it step 10).   A multithreaded application allows you to run several threads, each thread running in its own process.  So theoretically you can run step 1 in one thread and at the same time run step 2 in another thread.  At the same time you could run step 3 in its own thread, and even step 4 in its own thread.  Hence step 1, step 2, step 3, and step 4 would run concurrently.  Theoretically, if all four steps took about the same time, you could finish your program in a quarter of the time it takes to run a single thread (assuming you had a 4 processor machine).  So why isn't every program multithreaded?  Because along with speed, you face complexity.  Imagine if step 1 somehow depends on the information in step 2.  The program might not run correctly if step 1 finishes calculating before step 2 or visa versa.

An Unusual Analogy

Another way to think of multiple threading is to consider the human body.  Each one of the body's organs (heart, lungs, liver, brain) are all involved in processes.  Each process is running simultaneously.  Imagine if each organ ran as a step in a process: first the heart, then the brain, then the liver, then the lungs.  We would probably drop dead.   So the human body is like one big multithreaded application.  All organs are processes running simultaneously, and all of these processes depend upon one another.  All of these processes communicate through nerve signals, blood flow and chemical triggers.  As with all multithreaded applications, the human body is very complex.  If some processes don't get information from other processes, or certain processes slow down or speed up, we end up with a medical problem.  That's why (as with all multithreaded applications) these processes need to be synchronized properly in order to function normally.

When to Thread

Multiple threading is most often used in situations where you want programs to run more efficiently.  For example, let's say your Window Form program contains a method (call it method_A) inside it that takes more than a second to run and needs to run repetitively.  Well, if the entire program ran in a single thread,  you would notice times when button presses didn't work correctly, or your typing was a bit sluggish.  If method_A was computationally intensive enough, you might even notice certain parts of your Window Form not working at all.  This unacceptable program behavior is a sure sign that you need multithreading in your program.  Another common scenario where you would need threading is in a messaging system.  If you have numerous messages being sent into your application,  you need to capture them at the same time your main processing program is running and distribute them appropriately.  You can't efficiently capture a series of messages at the same time you are doing any heavy processing, because otherwise you may miss messages.  Multiple threading can also be used in an assembly line fashion where several processes run simultaneously.  For example once process collects data in a thread, one process filters the data, and one process matches the data against a database.  Each of these scenarios are common uses for multithreading and will significantly improve performance of similar applications running in a single thread.

When not to Thread

It is possible that when a beginning programmer first learns threading, they may be fascinated with the possibility of using threading in their program.  They may actually become thread-happy.  Let me elaborate:

Day 1) Programmer learns the that they can spawn a thread and begins creating a single new thread in their program, Cool!

Day 2) Programmer says,  " I can make this even more efficient by spawning other threads in parts of my program!"

Day 3)  P:  "Wow, I can even fork threads within threads and REALLY improve efficiency!!"

Day 4)  P: "I seem to be getting some odd results, but that's okay, we'll just ignore them for now."

Day 5)  "Hmmmm, sometimes my widgetX variable has a value,  but other times it never seems to get set at all, must be my computer isn't working, I'll just run the debugger".

Day 9)  "This darn (stronger language) program is jumping all over the place!!  I can't figure out what is going on!"

Week 2)  Sometimes the program just sits there and does absolutely nothing!  H-E-L-P!!!!!

Sound familiar?  Almost anyone who has attempted to design a multi-threaded program for the first time, even with good design knowledge of threading, has probably experienced at least 1 or 2 of these daily bullet points.  I am not insinuating that threading is a bad thing,   I'm just pointing out that in the process of creating threading efficiency in your programs, be very, very careful.   Because unlike a single threaded program, you are handling many processes at the same time, and multiple processes, with multiple dependent variables, can be very tricky to follow. Think of multithreading like you would think of juggling.  Juggling a single ball in your hand (although kind of boring)  is fairly simple.  However, if you are challenged to put two of those balls in the air, the task is a bit more difficult.  3, 4, and 5, balls are progressively more difficult.  As the ball count increases, you have a better and better chance of really dropping the ball.  Juggling a lot of balls at once requires knowledge, skill, and precise timing.  So does multiple threading.

 

Figure 1 - Multithreading is like juggling processes

Problems with Threading

If every process in your program was mutually exclusive - that is, no process depended in any way upon another, then multiple threading would be very easy and very few problems would occur.  Each process would run along in its own happy course and not bother the other processes.  However, when more than one process needs to read or write the memory used by other processes, problems can occur.  For example let's say there are two processes, process #1 and process #2.  Both processes share variable X.  If thread process #1 writes variable X with the value 5 first and thread process #2 writes variable X with value -3 next, the final value of X is -3.  However if process #2 writes variable X with value -3 first and then process #1 writes variable X with value 5, the final value of  X is 5.  So you see,  if the process that allows you to set X has no knowledge of process #1 or process #2, X can end up with different final values depending upon which thread got to X first.  In a single threaded program, there is no way this could happen, because everything follows in sequence.  In a single threaded program, since no processes are running in parallel,  X is always set by method #1 first, (if it is called first) and then set by method #2.  There are no surprises in a single threaded program, it's just step by step.    With a mulithreaded program, two threads can enter a piece of code at the same time, and wreak havoc on the results.  The problem with threads is that you need some way to control one thread accessing a shared piece of memory while another thread running at the same time is allowed to enter the same code and manipulate the shared data. 

Thread Safety

Imagine if every time you juggled three balls,  the ball in the air, by some freak of nature, was never allowed to reach your right hand until the ball already sitting in your right hand was thrown.  Boy, would juggling be a lot easier!  This is what thread safety is all about. In our program we force one thread to wait inside our code block while the other thread is finishing its business.  This activity, known as thread blocking or synchronizing threads, allows us to control the timing of simultaneous threads running inside our program.  In C# we lock on a particular part of memory (usually an instance of an object) and don't allow any thread to enter code of this object's memory until another thread is done using the object.  By now you are probably thirsting for a code example, so here you  go.

Let's take a look at a two-threaded scenario.   In our example, we will create two threads in C#: Thread 1 and Thread 2, both running in their own while loop.  The threads won't do anything useful, they will just print out a message saying which thread they are part of.  We will utilize a shared memory class member called _threadOutput.  _threadOutput will be assigned a message based upon the thread in which it is running.  Listing #1 shows the two threads contained in DisplayThread1 and DisplayThread2 respectively.

Listing 1 - Creating two threads sharing a common variable in memory

            // shared memory variable between the two threads
            // used to indicate which thread we are in
            private string _threadOutput = "";

            /// <summary>
            /// Thread 1: Loop continuously,
            /// Thread 1: Displays that we are in thread 1
            /// </summary>
            void DisplayThread1()
            {
                  while (_stopThreads == false)
                  {
                        Console.WriteLine("Display Thread 1");

                        // Assign the shared memory to a message about thread #1
                        _threadOutput = "Hello Thread1";


                        Thread.Sleep(1000);  // simulate a lot of processing
 

                        // tell the user what thread we are in thread #1, and display shared memory
                        Console.WriteLine("Thread 1 Output --> {0}", _threadOutput);

                  }
            }

 

            /// <summary>
            /// Thread 2: Loop continuously,

            /// Thread 2: Displays that we are in thread 2
            /// </summary>
            void DisplayThread2()
            {
                  while (_stopThreads == false)
                  {
                    Console.WriteLine("Display Thread 2");
 

                   // Assign the shared memory to a message about thread #2
                    _threadOutput = "Hello Thread2";


                    Thread.Sleep(1000);  // simulate a lot of processing

 
                  // tell the user we are in thread #2
                    Console.WriteLine("Thread 2 Output --> {0}", _threadOutput);

                  }
            }

 

            Class1()
            {
                  // construct two threads for our demonstration;
                  Thread thread1 = new Thread(new ThreadStart(DisplayThread1));
                  Thread thread2 = new Thread(new ThreadStart(DisplayThread2));

                  // start them
                  thread1.Start();
                  thread2.Start();
            }

The results of this code is shown in figure 2.  Look carefully at the results.  You will notice that the program gives some surprising output(if we were looking at this from a single-threaded mindset).  Although we clearly assigned _threadOutput to a string with a number corresponding to the thread to which it belongs, that is not what it looks like in the console:

Figure 2 - Unusual output from our two thread example.

We would expect to see the following from our code

 Thread  1 Output --> Hello Thread 1   and  Thread 2 Output --> Hello Thread 2,   but for the most part, the results are completely unpredictable. 

Sometimes we see   Thread  2 Output --> Hello Thread 1   and  Thread 1 Output --> Hello Thread 2.   The thread output does not match the code!  Even though, we look at the code and follow it with our eyes,  _threadOutput = "Hello Thread 2",   Sleep,    Write "Thread 2 -->  Hello Thread 2",  this sequence we expect does not necessarily produce the final result. 

Explanation

The reason we see the results we do is because in a multithreaded program such as this one, the code theoretically is executing the two methods DisplayThread1 and DisplayThread2, simultaneously. Each method shares the variable, _threadOutput.  So it is possible that although _threadOutput is assigned a value "Hello Thread1" in thread #1 and displays _threadOutput two lines later to the console, that somewhere in between the time thread #1 assigns it and displays it, thread #2 assigns _threadOutput the value  "Hello Thread2".  Not only are these strange results, possible, they are quite frequent as seen in the output shown in figure 2.  This painful threading problem is an all too common bug in thread programming known as a race condition.  This example is a very simple example of the well-known threading problem.  It is possible for this problem to be hidden from the programmer much more indirectly such as through referenced variables or collections pointing to thread-unsafe variables.   Although in figure 2 the symptoms are blatant, a race condition can appear much more rarely and be intermittent once a minute, once an hour, or appear three days later.  The race is probably the programmer's worst nightmare because of its infrequency and because it can be very very hard to reproduce.

Winning the Race

The best way to avoid race conditions is to write thread-safe code.  If your code is thread-safe,  you can prevent some nasty threading issues from cropping up.  There are several defenses for writing thread-safe code.  One is to share memory as little as possible.  If you create an instance of a class, and it runs in one thread,  and then you create another instance of the same class, and it runs in another thread, the classes are thread safe as long as they don't contain any static variables.  The two classes each create their own memory for their own fields, hence no shared memory.    If you do have static variables in your class or the instance of your class is shared by several other threads,  then you must find a way to make sure one thread cannot use the memory of that variable until the other class is done using it.    The way we prevent one thread from effecting the memory of the other class while one is occupied with that memory is called locking.  C# allows us to lock our code with either a Monitor class or a lock { }  construct.   (The lock construct actually internally implements the Monitor class through a try-finally block, but it hides these details from the programmer).  In our example in listing 1,  we can lock the sections of code from the point in which we populate the shared _threadOutput variable all the way to the actual output to the console.  We lock our critical section of code in both threads so we don't have a race in one or the other.  The quickest and dirtiest way to lock inside a method is to lock on the this pointer.  Locking on the this pointer will lock on the entire class instance, so any thread trying to modify a field of the class while inside the lock, will be blocked.  Blocking means that the thread trying to change the variable will sit and wait until the lock is released on the locked thread.   The thread is released from the lock upon reaching the last bracket in the lock { } construct.

Listing 2 - Synchronizing two Threads by locking them

           /// <summary>
           /// Thread 1, Displays that we are in thread 1 (locked)
            /// </summary>
            void DisplayThread1()
            {
                  while (_stopThreads == false)
                  {
                     
// lock on the current instance of the class for thread #1
                        lock (this)
                        {
                              Console.WriteLine("Display Thread 1");
                              _threadOutput = "Hello Thread1";
                              Thread.Sleep(1000);  // simulate a lot of processing
                              // tell the user what thread we are in thread #1
                              Console.WriteLine("Thread 1 Output --> {0}", _threadOutput);
                        }
// lock released  for thread #1 here
                  }
            }

 

           /// <summary>
           /// Thread 1, Displays that we are in thread 1 (locked)
            /// </summary>
            void DisplayThread2()
            {
                  while (_stopThreads == false)
                  {

                      // lock on the current instance of the class for thread #2
                        lock (this)
                        {
                              Console.WriteLine("Display Thread 2");
                              _threadOutput = "Hello Thread2";
                              Thread.Sleep(1000);  // simulate a lot of processing
                              // tell the user what thread we are in thread #1
                              Console.WriteLine("Thread 2 Output --> {0}", _threadOutput);
                        } 
// lock released  for thread #2 here
                  }
            }

 

The results of locking the two threads is shown in figure 3.  Note that all thread output is nicely synchronized.  You always get a result saying   Thread  1 Output --> Hello Thread 1   and  Thread 2 Output --> Hello Thread 2.   Note, however, that thread locking does come at a price.  When you lock a thread,  you force the other thread to wait until the lock is released.  In essence, you've slowed down the program, because while the other thread is waiting to use the shared memory, the first thread isn't doing anything in the program.  Therefore you need to use locks sparingly; don't just go and lock every method you have in your code if they are not involved in shared memory.  Also be careful when you use locks, because you don't want to get into the situation where thread #1 is waiting for a lock to be released by thread #2,  and thread #2 is waiting for a lock to be released by thread #1. When this situation happens, both threads are blocked and the program appears frozen.  This situation is known as deadlock and is almost as bad a situation as a race condition, because it can also happen at unpredictable, intermittent periods in the program. 

 

  Figure 3 - Synchronizing the dual thread program using locks

Alternative Solution

.NET provides many mechanisms to help you control threads.  Another way to keep a thread blocked while another thread is processing a piece of shared memory is to use an AutoResetEvent.  The AutoResetEvent class has two methods,  Set and WaitOne.  These two methods can be used together for controlling the blocking of a thread.  When an AutoResetEvent is initialized with false, the program will stop at the line of code that calls WaitOne until the Set method is called on the AutoResetEvent.  After the Set method is executed on the AutoResetEvent,  the thread becomes unblocked and is allowed to proceed past WaitOne.   The next time WaitOne is called, it has automatically been reset, so the program will again wait (block) at the line of code in which the WaitOne method is executing.  You can use this "stop and trigger" mechanism to block on one thread until another thread is ready to free the blocked thread by calling Set.  Listing 3 shows our same two threads using the AutoResetEvent to block each other while the block thread waits and the unblocked thread executes to display _threadOutput to the Console.  Initially, _blockThread1 is initialized to signal false, while _blockThread2  is initialized to signal true.  This means that _blockThread2 will be allowed to proceed through the WaitOne call the first time through the loop in DisplayThread_2,  while _blockThread1 will block on its WaitOne call in DisplayThread_1.   When the _blockThread2 reaches the end of the loop in thread 2, it signals _blockThread1 by calling Set in order to release thread 1 from its block.  Thread 2 then waits in its WaitOne call until Thread 1 reaches the end of its loop and calls Set on _blockThread2.  The Set called in Thread 1 releases the block on thread 2 and the process starts again.   Note that if we had set both AutoResetEvents (_blockThread1 and _blockThread2)  initially to signal false, then both threads would be waiting to proceed through the loop without any chance to trigger each other, and we would experience a deadlock. 

Listing 3 - Alternatively Blocking threads with the AutoResetEvent

            AutoResetEvent _blockThread1 = new AutoResetEvent(false);
            AutoResetEvent _blockThread2 = new AutoResetEvent(true);

            /// <summary>
            /// Thread 1, Displays that we are in thread 1
            /// </summary>
            void DisplayThread_1()
            {
                  while (_stopThreads == false)
                  {
                           // block thread 1  while the thread 2 is executing
                            _blockThread1.WaitOne(); 

                            // Set was called to free the block on thread 1, continue executing the code
                              Console.WriteLine("Display Thread 1");

                              _threadOutput = "Hello Thread 1";
                              Thread.Sleep(1000);  // simulate a lot of processing

 
                              // tell the user what thread we are in thread #1
                              Console.WriteLine("Thread 1 Output --> {0}", _threadOutput);

                            // finished executing the code in thread 1, so unblock thread 2
                              _blockThread2.Set();
                  }
            }

            /// <summary>
            /// Thread 2, Displays that we are in thread 2
            /// </summary>
            void DisplayThread_2()
            {
                  while (_stopThreads == false)
                  {
                        // block thread 2  while thread 1 is executing
                              _blockThread2.WaitOne(); 

                        // Set was called to free the block on thread 2, continue executing the code
                              Console.WriteLine("Display Thread 2");

                              _threadOutput = "Hello Thread 2";
                              Thread.Sleep(1000);  // simulate a lot of processing

                               // tell the user we are in thread #2
                              Console.WriteLine("Thread 2 Output --> {0}", _threadOutput); 

                        // finished executing the code in thread 2, so unblock thread 1
                            _blockThread1.Set();
                  }
            }

The output produced by listing 3  is the same output as our locking code, shown in figure 3,  but the AutoResetEvent gives us some more dynamic control over how one thread can notify another thread when the current thread is done processing.

Conclusion

As we are pushing the theoretical limits of microprocessor speed, technology needs to find new ways to be able to optimize speed and performance in computer technology.  With the invention of multiple processor chips and the inroads into parallel programming,  understanding multithreading can prepare you for the paradigm needed to handle these more recent technologies that will bring us the advantage we need to continue to challenge Moore's Law.   C# and .NET  give us the ability to support multithreading and parallel processing. If we understand how to use to utilize these tools skillfully, we can prepare for these hardware promises of the future  in our own day-to-day programming activities. In the meantime, sharpen your knowledge of threading, so you can .net the possibilities. 


Login to add your contents and source code to this article
 About the author
 
Mike Gold
Michael Gold is President of Microgold Software Inc., makers of the WithClass UML Tool. His company is a Microsoft VBA Partner and Borland Partner. Mike is a Microsoft MVP and founding member of C# Corner. He has a BSEE and MEng EE from Cornell University and has consulted for Chase Manhattan Bank, JP Morgan, Merrill Lynch, and Charles Schwab. Currently he is a senior developer at Finisar Corp. He has been involved in several .NET book projects, and is currently working on a book for using .NET with embedded systems. He can be reached at mike@c-sharpcorner.com
Looking for C# Consulting?
C# Consulting is founded in 2002 by the founders of C# Corner. Unlike a traditional consulting company, our consultants are well-known experts in .NET and many of them are MVPs, authors, and trainers. We specialize in Microsoft .NET development and utilize Agile Development and Extreme Programming practices to provide fast pace quick turnaround results. Our software development model is a mix of Agile Development, traditional SDLC, and Waterfall models.
Click here to learn more about C# Consulting.
 
Introducing MaxV - one click. infinite control. Hyper-V Hosting from MaximumASP.
Finally – a virtual platform that delivers next-generation Windows Server 2008 Hyper-V virtualization technology from a managed hosting partner you can truly depend on. Visit www.maximumasp.com/max for a FREE 30 day trial. Hurry offer ends soon. Climb aboard the MaxV platform and take advantage of High Availability, Intelligent Monitoring, Recurrent Backups, and Scalability – with no hassle or hidden fees. As a managed hosting partner focused solely on Microsoft technologies since 2000, MaximumASP is uniquely qualified to provide the superior support that our business is built on. Unparalleled expertise with Microsoft technologies lead to working directly with Microsoft as first to offer IIS 7 and SQL 2008 betas in a hosted environment; partnering in the Go Live Program for Hyper-V; and product co-launches built on WS 2008 with Hyper-V technology.
Dynamic PDF
ceTE software specializes in components for dynamic PDF generation and manipulation. The DynamicPDF™ product line allows you to dynamically generate PDF documents, merge PDF documents and new content to existing PDF documents from within your applications.
Go.NET
Build custom interactive diagrams, network, workflow editors, flowcharts, or software design tools. Includes many predefined kinds of nodes, links, and basic shapes. Supports layers, scrolling, zooming, selection, drag-and-drop, clipboard, in-place editing, tooltips, grids, printing, overview window, palette. 100% implemented in C# as a managed .NET Control. Document/View/Tool architecture with many properties&events. Optional automatic layout.
Dundas Software
Dundas Chart for .NET is the most advanced .NET charting package available today.  With an extremely complete feature set, elegant architecture and easy implementation, Dundas Chart can quickly add advanced Charting functionality to enhance and transform ASP.NET and Windows Forms applications.  Whether you are implementing charting into internal projects, or building applications for clients, Dundas Chart offers advanced technology and advanced results to get the most out of data.
Clickatell's SMS Gateway
Clickatell's Developer Solutions allow you to SMS enable any website or application via a range of API's. Learn More about our API connections.
Free access to .NET Memory Management video
Everything you need to know about Garbage Collection, Temporary Objects, Fragmentation, Finalization and common causes of memory leaks in .NET. Watch the video here.
Microsoft Visual Studio 2010 Professional
Microsoft Visual Studio 2010 Professional will launch on April 12, but you can beat the rush and secure your copy today by pre-ordering at the affordable estimated retail price of $549 (US). Pre-order now.
Nevron Chart for .NET 2010.1 Now Available
The leading .NET charting control now features PDF, Flash and Silverlight export, visualization of large datasets and more. Deliver true charting functionality to your BI, Scorecard, Presentation or Scientific apps. Download evaluation now.
Developer-Ready ASP.NET 2.0 Web Hosting with 3 MONTHS FREE
Now supporting .NET 3.0 Framework with Windows Workflow Foundation, Windows Communication Foundation (WCF), Windows Presentation Foundation (WPF), windows CardSpace (WCS)! Providing more flexibility for Developers with Web Services Support and a User/Permission Manger. Also supporting MS SQL 2005/2000 with Real-Time Backups, FREE Automated Attach .MDF Tool, FREE SQL Restore and Shrink SQL DB Tools, and SQL
 
   Print Read/Post comments Post a comment  Similar Articles  
   Email to a friend  Bookmark  Author's other articles  
Download Files:
multithreading.ZIP
 
 Post a Feedback, Comment, or Question about this article
Subject:  
Comment:  
Become a Sponsor
 Comments
Article by Nataraj On December 11, 2008
Excellent!!!!!
Reply | Email | Delete | Modify | 
Good job!!! by thaiasp On March 9, 2009
Your explanation is so clearly, simple and easy to understand. I've got the idea now. Thanks :)
Reply | Email | Delete | Modify | 
Parallel Framework (task parallelism) instead of plain threading by sjaak On March 23, 2009
Instead of using plain threading to develop concurrent software, please also consider task parallelism.

Please read the following articles:
 http://blog.rednael.com/2009/02/05/ParallelProgrammingUsingTheParallelFramework.aspx
 http://blog.rednael.com/2009/03/16/ParallelFrameworkDownloadAndSupport.aspx

These are articles about basic parallel programming and also contain references to other articles about the background of parallelism. Included are examples in C# .Net.
Also, it describes a lightweight parallel framework to work with tasks. Opposed to some other frameworks, this one is very light and very easy to use.

After reading these articles, you should be able to write code using parallelism.

Regards,
Martijn
Reply | Email | Delete | Modify | 
Very good article in simple words to understand by mahesh On January 10, 2010
Very nice article. I had been to interview. but cought out because i was not good with multi threading concept. This article is really good to know everything about threading....
Reply | Email | Delete | Modify | 
Some points... by Brian On February 4, 2010

Good article on multi-threading but a few points could be added to it that would make it clearer to the new to the subject Mike...

Threads are not processes but instead share the same process tree making the OS switching between them much quicker than between full processes e.g. they share the same memory and resources etc, they therefore inherit a lot of the basic process information.

Most professional application developers don't use multi-threading not just because of complexity but because threads are either not workable or they hinder the application e.g. makes things slower or use vast amount of memory/resources.

Before any multi-threading is done it is very important that an investigation into the benefits and pitfalls is covered but few who are new to the subject do this.

Pitfalls are many and just a few are listed below...

·        System flogging – heavily using the available resources of the whole system, memory hogging – amount and bus bandwidth, disk thrashing. Especially an issue if the app is being targeted at a shared server and not a home PC.

·        Shared memory bus means memory thrashing can happen where a single thread would not cause issues.

·        Shared caches in multi-core chips can be greatly flogged meaning once again that the single thread version runs quicker than the multi-thread one. E.g. a shared 8MB level 3 cache (i7) could make a single thread run very quick while the multi-cored version splits this cache between all the threads and may lead to an overall slowing down.

·        Disk thrashing in disk I/O based threads can be very high as the single thread version may do loads of sequential block reads but the multi-thread version can end up moving the physical read head about a large amount so for this reason it is important that the number of disks in the RAID array is taken into consideration.

·         Memory hogging – many threads can add a lot of memory to the application
footprint meaning that other processes have little or no memory to work with.

Threads overheads can also be very high with thread switching and locks meaning once again that the single thread main process runs faster than the multi-threaded version.


The number of threads should be adaptable so that the process checks before running how many threads are useful, e.g. checking number of cores and running less than that number of threads.

I have used multi-threading in the past but limited it to cases where it worked well e.g. graphics where the code was doing lots of floating point maths all within the CPU core cache footprint or for messaging servers where it is a necessary thing to do.

Even professional apps like SQL Server and Photoshop use it far less than expected and SQL has a limiting hint (MAXDOP) so you can stop thrashing if the query plan generated uses too much multi-threading and slows down the query.

Brian

Reply | Email | Delete | Modify | 
Re: Some points... by Antony On February 22, 2010
Excellent article; and thanks BRIAN for the relevant clarification too!!
Reply | Email | Delete | Modify | 
Re: Re: Some points... by Brian On February 23, 2010

Also what should be noted are the expected gains where an application lends itself to multi-threading...

On such applications as long as the memory bandwidth is not stressed you can expect a 40% to 60% gain for each doubling of cores used and a 20% to 30% gain if using hyper-threaded virtual cores.

So even for applications that suit multi-threading very well, the speed advantage is not as expected by the new to the subject.

It is often far better to spend the time optimising the single threaded code first to use the memory PreFetcher better and CPU Caching better as these tactics can usually see a 2 to 10 fold performance increase on high performance areas of the code than to look to multi-threading an existing single threaded application.

Reply | Email | Delete | Modify | 
Re: Re: Re: Some points... by Mike On March 1, 2010
Some great points Brian, especially the point that you do not get 1 to 1 gains by increasing cores (although you do see significant performance improvements).   I recently played around with PLINQ on my Quad Core, and you do see a big jump in performance (although certainly not a 4X gain).

-Mike
Reply | Email | Delete | Modify | 

 Hosted by MaximumASP  |  Found a broken link?  |  Contact Us  |  Terms & conditions  |  Privacy Policy  |  Site Map  |  Suggest an Idea  |  Media Kit
Current Version: 5.2009.6.2
 © 2010  contents copyright of their authors. Rest everything copyright Mindcracker. All rights reserved.