top of page

Hashmap in Java

Updated: Oct 16, 2023

Overview

HashMap in Java is a part of the collections framework, which is found in java.util package. It provides the basic implementation of Map interface of Java. It stores the data in a key-value mapping in which every key is mapped to exactly one value of any data type.

Keys should be unique as the key is used to retrieve the corresponding value from the map. Since Java 5, it is denoted as HashMap<K,V>, where K stands for Key and V for Value. In this article, we will see the uses of HashMap in Java, its internal working and examples.


Introduction to Java HashMap

HashMap in Java is a collection that implements Map interface. HashMap<K, V> stores the data in (Key, Value) pairs. Here, keys are unique identifiers used to associate each value on a map.

HashMap is unsynchronised, therefore it's faster and uses less memory than HashTable. Being unsynchronized means HashMap doesn’t guarantee any specific order of the elements.

Also, HashMap in Java allows to add null keys but there should be only one entry with null as key. However, there can be any number of entries with null as value.

We should choose HashMap over HashTable for an unsynchronized or single-threaded application. Although, since JDK 1.8 HashTable has been deprecated.

Syntax:

The declaration for java.util.HashMap class is as follows:

public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable, Serializable

The HashMap in Java implements Serializable, Cloneable, Map<K,V> interfaces and it extends AbstractMap<K,V> class.


Parameters:

The HashMap in Java takes two parameters which are as follows:

  1. K: It is the data type of keys maintained by the map. Keys should be unique.

  2. V: It is the data type of values maintained. Values need not be unique. It can be duplicate.

Basic Usage of HashMap in Java

HashMap in Java is the implementation of map, which is a key-value mapping. Now, one might question that if we simply want to add values, then why can't we use a list? Why do we need HashMap? The simple reason is performance.

To find a specific element in a list, the time complexity is O(n)O(n) for insertion and lookup and if the list is sorted, it will be O(logn)O(logn) on using binary search.

The advantage of HashMap is that the time complexity to insert and retrieve a value is O(1)O(1) on average, and space complexity is O(n)O(n). We will look at the performance of HashMap in Java later in the article.


Example: Create HashMap in Java

First, the java.util.HashMap package needs to be imported to create a HashMap in Java. After the import is done, we can create HashMap in Java as below:

HashMap<K, V> languages = new HashMap<>();

In the above code, a hashmap named languages is created. Here, K represents the key type and V represents the type of values such as below:

HashMap<String, Integer> languages = new HashMap<>();

Here, the type of key is String, and the type of value is Integer.

Example:

package p4n.in;
import java.util.HashMap;

class Main {
    public static void main(String[] args) {

        // create a hashmap
        HashMap<String, Integer> languages = new HashMap<>();

        // add elements to hashmap
        languages.put("Java", 8);
        languages.put("JavaScript", 1);
        languages.put("Python", 3);
        System.out.println("HashMap: " + languages);
    }
}

Output:

HashMap: {Java=8, JavaScript=1, Python=3}

In the above example, we have created a HashMap named languages. Here, we have used the put() method to add elements to the hashmap. We will learn more about the put() method later in this article.


Hierarchy of HashMap in Java


The hierarchy of HashMap is as follows:


The HashMap in Java implements Serializable, Cloneable, Map interfaces. It extends AbstractMap class.


Constructors in HashMap Java


HashMap in Java has 4 constructors, and each one has public access modifier. The constructors are as follows:

  1. HashMap()

  2. HashMap(int initialCapacity)

  3. HashMap(int initialCapacity, float loadFactor)

  4. HashMap(Map map)

Before discussing the constructors in detail, let's take a deep dive into a few other terms for better understanding:

  • Hashing: HashMap in Java works on the principle of hashing — an algorithm to map object data to some representative integer values.

Hashing is a process of converting an object into integer form by using the method hashCode(). hashCode() method of object class returns the memory reference of object in integer form. So the Hash function is applied to the key object to calculate the index of the bucket in order to store and retrieve any key-value pair.

HashMap<K,V> manages an array of Node<K,V> objects that will be replaced by another larger array if all of its elements has been assigned value. Node<K,V> class consists of 4 fields. The next field is a reference to the next Node object.

It may not be the next element in the array. HashMap ensures those Node(s) having the same hashcode will have consecutive references. This helps it quickly find all Node(s) with the same specified hashcode.


  • Capacity: Capacity is the number of buckets in the HashMap. The initial capacity is the capacity at which the Map is created. The initial default capacity of the HashMap is 16`.

  • Load Factor: The capacity is expanded as the number of elements in the HashMap increases. The load factor is the measure that decides when to increase the capacity of the Map. The default load factor is 75% of the capacity.

  • Threshold: The threshold of a HashMap in Java is approximately the product of the current capacity and load factor.

  • Rehashing: Rehashing is the process of re-calculating the hash code of already stored entries. When the number of entries in the hash table exceeds the threshold value, the Map is rehashed so that it has approximately twice the number of buckets as before.

  • Collision: A collision occurs when a hash function returns the same bucket location for two different keys.

Let's suppose a HashMap is created with the initial default capacity of 16 and the default load factor of 0.75. So, the threshold is 16 * 0.75 = 12, which means that it will increase the capacity from 16 to 32 after the 12th entry (key-value pair) is added. This will be done by rehashing.


Constructor 1: HashMap()

This is the default constructor, which creates a HashMap instance with a default initial capacity of 16 and a load factor of 0.75.

  • Syntax:

HashMap<K, V> hm = new HashMap<K, V>();
  • Example:

package p4n.in;
import java.io.*;
import java.util.HashMap;

class p4np4n{

	public static void main(String args[])
	{
		// Initializing the HashMap
               // No need to mention the Generic type twice
		HashMap<Integer, String> hMap1 = new HashMap<>();
		HashMap<Integer, String> hMap2 = new HashMap<>();

		// Adding elements using put method
		hMap1.put(1, "Java");
		hMap1.put(2, "C");
		hMap1.put(3, "Python");

		hMap2.put(4, "Javascript");
		hMap2.put(5, "Kotlin");
		hMap2.put(6, "Go");

		// Printing HashMap 1
		System.out.println("HashMap hMap1 : "
						+ hMap1);

		// Printing HashMap 2
		System.out.println("HashMap hMap2 : "
						+ hMap2);
	}
}

Output:

HashMap hMap1 : {1=Java, 2=C, 3=Python}
HashMap hMap2 : {4=Javascript, 5=Kotlin, 6=Go}

In the above example, we have created two HashMaps hMap1 and hMap2, using the default HashMap() constructor and adding 3 values in each HashMap using put() method. Then we have printed both the HashMaps.


Constructor 2: HashMap(int initialCapacity)


This constructor creates an instance of HashMap with a specified initial capacity and a load factor of 0.75.

  • Syntax:

HashMap<K, V> hm = new HashMap<K, V>(int initialCapacity);
  • Example:

package p4n.in;
import java.io.*;
import java.util.HashMap;

class p4n{

	public static void main(String args[])
	{
		// Initializing the HashMap
		HashMap<Integer, String> hMap1 = new HashMap<>(10);
		HashMap<Integer, String> hMap2 = new HashMap<>(5);

		// Adding elements
		hMap1.put(1, "Java");
		hMap1.put(2, "C");
		hMap1.put(3, "Python");

		hMap2.put(4, "Javascript");
		hMap2.put(5, "Kotlin");
		hMap2.put(6, "Go");

		// Printing HashMap 1 elements
		System.out.println("HashMap hMap1 : "
						+ hMap1);

		// Printing HashMap 2 elements
		System.out.println("HashMap hMap2 : "
						+ hMap2);
	}
}

Output:

HashMap hMap1 : {1=Java, 2=C, 3=Python}
HashMap hMap2 : {4=Javascript, 5=Kotlin, 6=Go}

In the above example, we have created two HashMaps with different intial capacities. hMap1 with 10 and hMap2 with 5 capacity. If the threshold value is exceeded then the capacity of hMap1 and hMap2 will increase to 20 and 10, respectively (approx. twice).


Constructor 3: HashMap(int initialCapacity, float loadFactor)


This constructor creates an instance of HashMap with a specified initial capacity and specified load factor.

  • Syntax:

HashMap<K, V> hm = new HashMap<K, V>(int initialCapacity, int  loadFactor);
  • Example:


package p4n.in;
import java.io.*;
import java.util.HashMap;

class p4n{

	public static void main(String args[])
	{
		// Initializing the HashMap
		HashMap<Integer, String> hMap1 = new HashMap<>(10, 0.75f);
		HashMap<Integer, String> hMap2 = new HashMap<>(5, 0.5f);

		// Adding elements
		hMap1.put(1, "Java");
		hMap1.put(2, "C");
		hMap1.put(3, "Python");

		hMap2.put(4, "Javascript");
		hMap2.put(5, "Kotlin");
		hMap2.put(6, "Go");

		// Printing HashMap 1 elements
		System.out.println("HashMap hMap1 : "
						+ hMap1);

		// Printing HashMap 2 elements
		System.out.println("HashMap hMap2 : "
						+ hMap2);
	}
}

Output:

HashMap hMap1 : {1=Java, 2=C, 3=Python}
HashMap hMap2 : {4=Javascript, 5=Kotlin, 6=Go}

In the above example we have created two HashMaps with different initial capacity and load factors. hMap1 with a capacity of 10 and load factor of 0.75, whereas hMap2 with a capacity of 5 and a load factor of 0.5.

The default load factor is 0.75 but here we are giving values as required. So the threshold value for both maps would be:

For hMap1: Threshold1=(10∗0.75)=7.5Threshold1=(10∗0.75)=7.5 For hMap2: Threshold2=(5∗0.5)=2.5Threshold2=(5∗0.5)=2.5 So if the respected threshold values exceed the capacity will be increased for the particular HashMap.


Constructor 4: HashMap(Map map)

This creates a HashMap instance with the same mappings as the specified map.

  • Syntax:

HashMap<K, V> hm = new HashMap<K, V>(Map map);
  • Example:


package p4n.in;
import java.io.*;
import java.util.HashMap;

class p4n{

	public static void main(String args[])
	{
		// Initializing the HashMap
		HashMap<Integer, String> hMap1 = new HashMap<>();

		// Adding elements
		hMap1.put(1, "Java");
		hMap1.put(2, "C");
		hMap1.put(3, "Python");
        
        //Initializing hMap2 with hMap1 values
        HashMap<Integer, String> hMap2 = new HashMap<>(hMap1);

		// Printing HashMap 1 elements
		System.out.println("HashMap hMap1 : "
						+ hMap1);

		// Printing HashMap 2 elements
		System.out.println("HashMap hMap2 : "
						+ hMap2);
	}
}

Output:

HashMap hMap1 : {1=Java, 2=C, 3=Python}
HashMap hMap2 : {1=Java, 2=C, 3=Python}

In the above example, we have created a HashMap hMap1 and then hMap2 by initializing the values of hMap1 to hMap2. So the hMap1 and hMap2 will have the same values, capacity and load factor.


How to Perform Various Operations on HashMap in Java?


1. Adding Elements

We can add an element to the HashMap in Java by using the put() method. However, the insertion order will not be carried out sequentially. Internally, a separate hash is generated for every element. Based on that hash the elements are indexed for more efficiency.


package p4n.in;
import java.io.*;
import java.util.HashMap;

class AddingElements {

	public static void main(String args[])
	{
		// Initializing the HashMap
               // No need to mention the Generic type twice
		HashMap<Integer, String> hMap1 = new HashMap<>();
		HashMap<Integer, String> hMap2 = new HashMap<>();

		// Adding elements using the put method
		hMap1.put(1, "p4n");
		hMap1.put(2, "Topics");

		hMap2.put(4, "p4n");
		hMap2.put(5, "Topics");

		// Printing HashMap 1
		System.out.println("HashMap hMap1 : "
						+ hMap1);

		// Printing HashMap 2
		System.out.println("HashMap hMap2 : "
						+ hMap2);
	}
}

Output:

HashMap hMap1 : {1=p4n, 2=Topics}
HashMap hMap2 : {4=p4n, 5=Topics}


2. Changing Elements

To change the element after adding, we can again use put() method by adding the element with the updated value for the key for which we want to change the value. This will replace the existing value as the elements in the map are indexed using the keys.


package p4n.in;
import java.io.*;
import java.util.HashMap;
class ChangeElements {
	public static void main(String args[])
	{

		// Initialization of a HashMap
		HashMap<Integer, String> hMap
			= new HashMap<Integer, String>();

		// Changing the Value using put method
		hMap.put(1, "p4n");
		hMap.put(2, "p4n");
		hMap.put(3, "Article");

		System.out.println("Initial Map " + hMap);

		hMap.put(2, "Topics");
p4n
		System.out.println("Updated Map " + hMap);
	}
}

Output:

Initial Map {1=p4n, 2=p4n, 3=Article}
Updated Map {1=p4n, 2=Topics, 3=Article}


3. Removing Element

The remove() method is used to remove the elements from the map. We need to pass the key in the method and then the particular key-value pair will be removed if it's present in the map.

package p4n.in;
import java.io.*;
import java.util.HashMap;
class RemoveElements {
	public static void main(String args[])
	{

		// Initialization of a HashMap
		HashMap<Integer, String> hMap
			= new HashMap<Integer, String>();

		// Changing the Value using put method
		hMap.put(1, "p4n");
		hMap.put(2, "codeswithpankaj");
		hMap.put(3, "Article");

		System.out.println("Initial Map :" + hMap);
        
                //Removing element with remove() method
                by passing the key
		hMap.remove(3);

		System.out.println("Updated Map :" + hMap);
	}
}

Output:

Initial Map : {1=p4n, 2=codeswithpankaj, 3=Article}
Updated Map : {1=p4n, 2=codeswithpankaj}


4. Traversal of HashMap

To traverse over any structure of the Collection Framework (like HashMap in Java), we can use the iterator interface. Using Java for-each loop, we can iterate through each entry of the HashMap. Iterators work with one type of data but in Map, we have two types- Key and Value. So we use Entry<K,V> to resolve this into a compatible format.


package p4n.in;
import java.util.HashMap;
import java.util.Map;

public class TraversalOfHshMap {
	public static void main(String[] args)
	{
		// Initialization of a HashMap
		HashMap<String, Integer> map = new HashMap<>();

		// Add elements
		map.put("Apple", 100);
		map.put("Orange", 70);
		map.put("Banana", 50);

		// Iterating the map using for-each loop
		for (HashMap.Entry<String, Integer> m : map.entrySet())
			System.out.println("Key: " + m.getKey()
							+ " Value: " + m.getValue());
	}
}

Output:

Key: Apple Value: 100
Key: Orange Value: 70
Key: Banana Value: 50

In the above example, we have used HashMap.Entry(). Entry is an inner class in whatever Map implementation we are using (here, in this case it's HashMap). HashMap.Entry is a key and its value combined into one class. This allows us to iterate over HashMap.entrySet() instead of having to iterate over HashMap.keySet(), then getting the value for each key. So, the HashMap.entrySet() method returns a collection-view (elements) of the HashMap.


Features of HashMap in Java

  • HashMap in Java uses the technique of Hashing. That's why it is called HashMap. Hashing function maps a big number or string to a small integer that can be used as an index. A shorter value helps in indexing and faster searches.

  • HashMap in Java is found in the java.util package.

  • In HashMap, the keys should be unique but values can be duplicated. It means that an X key can't contain more than one value that is X mapped to only one value. But other Y, Z keys can contain duplicate or the same values already mapped to X key.

  • HashMap in Java can also have null key and values. But there could be only one null key. However, there could be any number of null values corresponding to different keys.

  • Unlike HashTable, HashMap is not synchronized. It is an unordered collection that doesn't guarantee any specific order of the elements.

  • To retrieve any value from the HashMap, one should know the associated key.

  • HashMap extends an abstract class AbstractMap and implements Cloneable and Serializable interfaces.

How does HashMap Work Internally?

As we have discussed, HashMap in Java stores the elements using key-value mapping where we can retrieve an element from the HashMap by its key. The key-value pairs are stored as instances of inner class HashMap. The entry which has key-value mapping stored as attributes where the key has been marked as final (immutable).

Internally, the HashMap is an array of nodes. HashMap makes use of array and LinkedList for storing key-value pairs.

Given below is a structure of a node of HashMap that is programmatically represented as a class.


As seen from the node representation above, a node has a structure similar to a linked list node. An array of these nodes is called Bucket. HashMap stores elements in buckets and the number of buckets is called capacity.

When we add a value to the map, the bucket in which the value will be stored is determined by the key's hashCode() method. Basically, a hash value is calculated using the key’s hash code. This hash value is used to calculate the index in the array for storing Entry object.

When we try to retrieve the value, the bucket is calculated by the HashMap in the same way – using hashCode(), which actually calculates the first index location. Then in that bucket, it iterates through the objects found and uses the key's equals() method to find the exact match.

To avoid having many buckets with multiple values, the capacity is doubled if 75% (the load factor) of the buckets become non-empty. The default value for the load factor is 75%, and the default initial capacity is 16. Both can be set in the constructor as discussed above.


Performance of HashMap

The performance of a HashMap is affected by two parameters:

  • Initial Capacity: The initial capacity is simply the capacity (number of buckets) during creation.

  • Load Factor: The load factor or LF, is a measure of how full the hash map should be after adding some values before it is resized.

The initial default capacity is 16 and the default load factor is 0.75. We can create a HashMap with custom values as well.

However, for custom values, we should understand the performance implications. When the number of hash map entries exceeds the product of LF and capacity, then rehashing occurs i.e. another `internal array is created with twice the size of the initial one, and all entries are moved over to new bucket locations in the new array.

A low initial capacity reduces space cost but increases the frequency of rehashing. Rehashing is obviously a very expensive process. So if we have many entries, we should set a considerably high initial capacity.

But if we set the initial capacity too high, we will pay the cost in iteration time. So a high initial capacity is good for a large number of entries coupled with little to no iteration. A low initial capacity is good for a few entries with a lot of iteration.

Inserting a value into a HashMap takes, on the average case, O(1)O(1) time. The hash function is computed, the bucket is chosen, and then the item is inserted.

In the worst-case scenario, all of the elements will have hashed to the same value, which means either the entire bucket list must be traversed or, in the case of open addressing, the entire map must be probed until an empty spot is found. Therefore, in the worst case, insertion takes O(n) time.


Note: From Java 8 onward, Java has started using Self-Balancing BST instead of a linked list for chaining. The advantage of self-balancing BST is, in the worst case (when every key maps to the same slot) search time is O(Log n).



Related Posts

See All

Java Date and Time API Tutorial

Welcome to Code with Pankaj! In this tutorial, we'll explore the Java Date and Time API, introduced in Java 8. This API provides a...

Σχόλια


bottom of page