As per the following link document: Java HashMap Implementation
I'm confused with the implementation of HashMap
(or rather, an enhancement in HashMap
). My queries are:
Firstly
static final int TREEIFY_THRESHOLD = 8;
static final int UNTREEIFY_THRESHOLD = 6;
static final int MIN_TREEIFY_CAPACITY = 64;
Why and how are these constants used? I want some clear examples for this. How they are achieving a performance gain with this?
Secondly
If you see the source code of HashMap
in JDK, you will find the following static inner class:
static final class TreeNode<K, V> extends java.util.LinkedHashMap.Entry<K, V> {
HashMap.TreeNode<K, V> parent;
HashMap.TreeNode<K, V> left;
HashMap.TreeNode<K, V> right;
HashMap.TreeNode<K, V> prev;
boolean red;
TreeNode(int arg0, K arg1, V arg2, HashMap.Node<K, V> arg3) {
super(arg0, arg1, arg2, arg3);
}
final HashMap.TreeNode<K, V> root() {
HashMap.TreeNode arg0 = this;
while (true) {
HashMap.TreeNode arg1 = arg0.parent;
if (arg0.parent == null) {
return arg0;
}
arg0 = arg1;
}
}
//...
}
How is it used? I just want an explanation of the algorithm.
To put it simpler (as much as I could simpler) + some more details.
These properties depend on a lot of internal things that would be very cool to understand - before moving to them directly.
TREEIFY_THRESHOLD -> when a single bucket reaches this (and the total number exceeds
MIN_TREEIFY_CAPACITY
), it is transformed into a perfectly balanced red/black tree node. Why? Because of search speed. Think about it in a different way:Some intro for the next topic. Why is the number of bins/buckets always a power of two? At least two reasons: faster than modulo operation and modulo on negative numbers will be negative. And you can't put an Entry into a "negative" bucket:
Instead there is a nice trick used instead of modulo:
That is semantically the same as modulo operation. It will keep the lower bits. This has an interesting consequence when you do:
This is where multiplying the buckets comes into play. Under certain conditions (would take a lot of time to explain in exact details), buckets are doubled in size. Why? When buckets are doubled in size, there is one more bit coming into play.
As such this process is called re-hashing. This might get slow. That is (for people who care) as HashMap is "joked" as: fast, fast, fast, slooow. There are other implementations - search pauseless hashmap...
Now UNTREEIFY_THRESHOLD comes into play after re-hashing. At that point, some entries might move from this bins to others (they add one more bit to the
(n-1)&hash
computation - and as such might move to other buckets) and it might reach thisUNTREEIFY_THRESHOLD
. At this point it does not pay off to keep the bin asred-black tree node
, but as aLinkedList
instead, likeMIN_TREEIFY_CAPACITY is the minimum number of buckets before a certain bucket is transformed into a Tree.
You would need to visualize it: say there is a Class Key with only hashCode() function overridden to always return same value
and then somewhere else, I am inserting 9 entries into a HashMap with all keys being instances of this class. e.g.
Tree traversal is faster {O(log n)} than LinkedList {O(n)} and as n grows, the difference becomes more significant.
The change in HashMap implementation was was added with JEP-180. The purpose was to:
However pure performance is not the only gain. It will also prevent HashDoS attack, in case a hash map is used to store user input, because the red-black tree that is used to store data in the bucket has worst case insertion complexity in O(log n). The tree is used after a certain criteria is met - see Eugene's answer.
TreeNode
is an alternative way to store the entries that belong to a single bin of theHashMap
. In older implementations the entries of a bin were stored in a linked list. In Java 8, if the number of entries in a bin passed a threshold (TREEIFY_THRESHOLD
), they are stored in a tree structure instead of the original linked list. This is an optimization.From the implementation:
HashMap
contains a certain number of buckets. It useshashCode
to determine which bucket to put these into. For simplicity's sake imagine it as a modulus.If our hashcode is 123456 and we have 4 buckets,
123456 % 4 = 0
so the item goes in the first bucket, Bucket 1.If our hashcode function is good, it will provide an even distribution so all the buckets will be used somewhat equally. In this case, the bucket uses a linked list to store the values.
But you can't rely on people to implement good hash functions. People will often write poor hash functions which will result in a non-even distribution.
The less even this distribution is, the further we're moving from O(1) operations and the closer we're moving towards O(n) operations.
The implementation of Hashmap tries to mitigate this by organising some buckets into trees rather than linked lists if the buckets becomes too large. This is what
TREEIFY_THRESHOLD = 8
is for. If a bucket contains more than eight items, it should become a tree.This tree is a Red-Black tree. It is first sorted by hash code. If the hash codes are the same, it uses the
compareTo
method ofComparable
if the objects implement that interface, else the identity hash code.If entries are removed from the map, the number of entries in the bucket might reduce such that this tree structure is no longer necessary. That's what the
UNTREEIFY_THRESHOLD = 6
is for. If the number of elements in a bucket drops below six, we might as well go back to using a linked list.Finally, there is the
MIN_TREEIFY_CAPACITY = 64
.When a hash map grows in size, it automatically resizes itself to have more buckets. If we have a small hash map, the likelihood of us getting very full buckets is quite high, because we don't that have many different buckets to put stuff into. It's much better to have a bigger hash map, with more buckets that are less full. This constant basically says not to start making buckets into trees if our hash map is very small - it should resize to be larger first instead.
To answer your question about the performance gain, these optimisations were added to improve the worst case. I'm only speculating but you would probably only see a noticeable performance improvement because of these optimisations if your
hashCode
function was not very good.Images are mine (thanks MSPaint). Reuse them however you like.