I'm working with a list that is shared among many threads.
I believe that to a good performance in this case, it will be good to use InterlockedExchange function to insert data in this list, but I have some doubts.
If a thread tries to read a variable that is being written by another thread with InterlockedExchange, What will be the reaction? The thread that is reading the variable will wait for the completion of writing or will continue running and can not read the variable?
is necessary to use InterlockedExchange to read the variable when it is being written with interlockedechange?
How to test this function to try to know what will be the reaction to a multiple access to a shared variable between threads?
The short answer is that
InterlockedExchange
is unlikely to help you. But it's important to understand why this is the case so you'll be able to have a better understanding of the situations where it may help.Why is concurrent reading AND writing a problem?
First note that any number of threads may read the same data concurrently without concern. We only start to worry as soon as one thread is able to write concurrently. Now, there's nothing inherently risky with writing itself. The problem comes in when even the slightest bit of decision logic is applied to the shared data being used concurrently.
For example, you might have a list. One thread (the writer) deletes an element from the list, while other threads are reading elements in the list. If you are unlucky with the timing of your operations, a reader might confirm an element exists at the same time the writer deletes it. And then when the reader tries to use the no longer valid element, you get Access Violations at best, and data corruption at worst.
How to protect the data
The simplest way to protect the data (without architectural changes) is usually to introduce some form locking/blocking mechanism. Before one thread starts a critical operation, it says: "I'm busy with the shared data, all other threads must wait until I'm done if they want to use the data."
Note that the simplistic approach does introduce other problems:
(NOTE: There are many alternate ways you could protect your data, but that's beyond the scope of this post.)
InterlockedExchange
This routine and its other
Interlocked***
siblings provide a simplified locking mechanism over the steps of a basic, generic write operation. NOTE It's still a locking mechanism as described above, and shares most of its problems.The two-step operation that
InterlockedExchange
protects is:The reason this might need protection is that if two threads Exchange a shared value concurrently, there is a possibility of inconsistent behaviour.
E.g. Given initial value is A. Thread #1 exchanges setting the value to B. Thread #2 exchanges setting the value to C. If the threads run concurrently with #1 processing fractionally sooner than #2, there are 2 possible results.
A <-- Discrepancy
.This won't always be a problem, but in some cases it might be. In which case
InterlockedExchange
serves as the simplest lock based protection mechanism.Why InterlockedExchange is unlikely to work for you
You said your data was in a list, but didn't state what kind of list; so I'm assuming a standard Delphi
TList
. Inserting an item into a list is not a simple operation internally.NB! Note: Even if you are using a list data structure that could itself benefit from
InterlockedExchange
, there are still other problems you need to be aware of.You have two sets of data here. There is the internal data of the list structure (you might not think of it as such, but it is there). And then there's the data of your actual records.
Even after protecting your list structure, if you have any threads that can concurrently update your records, you have a potential problem. You need to protect your data - not just the act of adding it to a collection. This means you might need a locking mechanism that spans both sets of data; and there's no chance that any of the basic, generic
Interlocked***
routines will achieve that.