Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Is the order of an unordered_map deterministic?

I'm wondering if there's any guarantee that the order of an unordered_map will always be the same across all CPUs, threads, etc.

I realize there may be no obvious pattern to the particular order itself (hence, 'unordered' map), but if I run my process on another machine, or multiple times in succession, or on different threads, will the order of inserted items always be the same if the hash functions and insertion order remains the same? In other words, if my code doesn't change, will every execution of my process result in the elements of the map being in the same order?

I've run a few tests, and the order of items after insertion seems to be the same each time, but that could merely be a fluke, and I only have this single machine to test on. I need to know whether the order can be influenced by any other factors such as CPU/memory architecture, OS (Windows 8 vs Windows 10), etc.

like image 237
Tyson Avatar asked Oct 16 '25 18:10

Tyson


1 Answers

TL;DR: It can be done, but I wouldn't recommend it. Use another data structure if you can; your own hash table, a "treap", a flat array, or something.

The order of items in an std::unordered_map (or set) is more dependent on the standard library implementation than hardware/CPU/etc.

For that reason, if you use the same library implementation across different hardware, and provide your own hash function (to make sure it's not randomized across runs - to combat DoS attacks against your data structures,) or otherwise hardware- or OS-dependent, then you should be fine.

However, if you are looking for a guarantee in the standard, you won't find any. The only relevant guarantee is that in the same instance of the object, the same key will hash to the same bucket. I don't think there is a guarantee even for different instances of the map, and I know from (painful) personal experience that there is no consistency across different runs of the application.

But not all hope is lost! If you stick to the same implementation of unordered_map, and use your own hash functions, and take a look at the implementation to make sure there are no hidden surprises (any dependency on hardware/OS/time/RNG/etc. should be relatively easy to spot) you can manage it.

Note that since you seem to be on Windows and probably using MSVC, the default unordered_map with the default hashing algorithm is not at all order-consistent across runs of the same compiled binary (at least it wasn't in 2013/2015 IIRC.)

Another thing to keep in mind is that if you are serious about consistency, you have to make sure you link against the CRT statically. If you link against the DLL version, some future patch/update can change the behavior for your application after you release it.

like image 152
yzt Avatar answered Oct 18 '25 11:10

yzt