- Multithreading with C# Cookbook(Second Edition)
- Eugene Agafonov
- 437字
- 2021-07-09 19:35:37
Performing basic atomic operations
This recipe will show you how to perform basic atomic operations on an object to prevent the race condition without blocking threads.
Getting ready
To step through this recipe, you will need Visual Studio 2015. There are no other prerequisites. The source code for this recipe can be found at BookSamples\Chapter2\Recipe1
.
How to do it...
To understand basic atomic operations, perform the following steps:
- Start Visual Studio 2015. Create a new C# console application project.
- In the
Program.cs
file, add the followingusing
directives:using System; using System.Threading; using static System.Console;
- Below the
Main
method, add the following code snippet:static void TestCounter(CounterBase c) { for (int i = 0; i < 100000; i++) { c.Increment(); c.Decrement(); } } class Counter : CounterBase { private int _count; public int Count => _count; public override void Increment() { _count++; } public override void Decrement() { _count--; } } class CounterNoLock : CounterBase { private int _count; public int Count => _count; public override void Increment() { Interlocked.Increment(ref _count); } public override void Decrement() { Interlocked.Decrement(ref _count); } } abstract class CounterBase { public abstract void Increment(); public abstract void Decrement(); }
- Inside the
Main
method, add the following code snippet:WriteLine("Incorrect counter"); var c = new Counter(); var t1 = new Thread(() => TestCounter(c)); var t2 = new Thread(() => TestCounter(c)); var t3 = new Thread(() => TestCounter(c)); t1.Start(); t2.Start(); t3.Start(); t1.Join(); t2.Join(); t3.Join(); WriteLine($"Total count: {c.Count}"); WriteLine("--------------------------"); WriteLine("Correct counter"); var c1 = new CounterNoLock(); t1 = new Thread(() => TestCounter(c1)); t2 = new Thread(() => TestCounter(c1)); t3 = new Thread(() => TestCounter(c1)); t1.Start(); t2.Start(); t3.Start(); t1.Join(); t2.Join(); t3.Join(); WriteLine($"Total count: {c1.Count}");
- Run the program.
How it works...
When the program runs, it creates three threads that will execute a code in the TestCounter
method. This method runs a sequence of increment/decrement operations on an object. Initially, the Counter
object is not thread-safe and we get a race condition here. So, in the first case, a counter value is not deterministic. We could get a zero value; however, if you run the program several times, you will eventually get some incorrect nonzero result.
In Chapter 1, Threading Basics, we resolved this problem by locking our object, causing other threads to be blocked while one thread gets the old counter value and then computes and assigns a new value to the counter. However, if we execute this operation in such a way, it cannot be stopped midway, we would achieve the proper result without any locking, and this is possible with the help of the Interlocked
construct. It provides the Increment
, Decrement
, and Add
atomic methods for basic math, and it helps us to write the Counter
class without the use of locking.
- Web Scraping with Python
- SQL Server 2012數據庫技術及應用(微課版·第5版)
- HoloLens Beginner's Guide
- CentOS 7 Server Deployment Cookbook
- PowerCLI Cookbook
- 深入淺出Spring Boot 2.x
- Java Web及其框架技術
- 小程序,巧運營:微信小程序運營招式大全
- Spring Boot企業級項目開發實戰
- Learning PHP 7
- Java網絡編程實戰
- 平面設計經典案例教程:CorelDRAW X6
- 計算機應用基礎(第二版)
- C# 7.1 and .NET Core 2.0:Modern Cross-Platform Development(Third Edition)
- 啊哈C語言?。哼壿嫷奶魬穑ㄐ抻啺妫?/a>