Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Memory overhead of Java HashMap compared to ArrayList

I am wondering what is the memory overhead of java HashMap compared to ArrayList?

Update:

I would like to improve the speed for searching for specific values of a big pack (6 Millions+) of identical objects.

Thus, I am thinking about using one or several HashMap instead of using ArrayList. But I am wondering what is the overhead of HashMap.

As far as i understand, the key is not stored, only the hash of the key, so it should be something like size of the hash of the object + one pointer.

But what hash function is used? Is it the one offered by Object or another one?

like image 603
elhoim Avatar asked Oct 06 '09 16:10

elhoim


People also ask

How much memory does a HashMap use Java?

A HashMap. Entry is 24 Bytes, not 16, for example. For many cases, this adds up to an enormous amount of memory wasted. For example, a HashMap<Integer, Double> needs about 100 Bytes per stored value due to boxing, with 12 bytes of actual data, and 88 bytes overhead.

Does HashMap take more space than array?

Well here are the facts. So if you were actually limited by finding a contiguous space in memory, HashMap's benefit is only that the array may be smaller. Objects take up extra space. As I remember, it's typically 8 or 16 bytes minimum depending on whether it's a 32- or 64-bit system.


2 Answers

If you're comparing HashMap with ArrayList, I presume you're doing some sort of searching/indexing of the ArrayList, such as binary search or custom hash table...? Because a .get(key) thru 6 million entries would be infeasible using a linear search.

Using that assumption, I've done some empirical tests and come up with the conclusion that "You can store 2.5 times as many small objects in the same amount of RAM if you use ArrayList with binary search or custom hash map implementation, versus HashMap". My test was based on small objects containing only 3 fields, of which one is the key, and the key is an integer. I used a 32bit jdk 1.6. See below for caveats on this figure of "2.5".

The key things to note are:

(a) it's not the space required for references or "load factor" that kills you, but rather the overhead required for object creation. If the key is a primitive type, or a combination of 2 or more primitive or reference values, then each key will require its own object, which carries an overhead of 8 bytes.

(b) In my experience you usually need the key as part of the value, (e.g. to store customer records, indexed by customer id, you still want the customer id as part of the Customer object). This means it is IMO somewhat wasteful that a HashMap separately stores references to keys and values.

Caveats:

  1. The most common type used for HashMap keys is String. The object creation overhead doesn't apply here so the difference would be less.

  2. I got a figure of 2.8, being 8880502 entries inserted into the ArrayList compared with 3148004 into the HashMap on -Xmx256M JVM, but my ArrayList load factor was 80% and my objects were quite small - 12 bytes plus 8 byte object overhead.

  3. My figure, and my implementation, requires that the key is contained within the value, otherwise I'd have the same problem with object creation overhead and it would be just another implementation of HashMap.

My code:

public class Payload {     int key,b,c;     Payload(int _key) { key = _key; } }   import org.junit.Test;  import java.util.HashMap; import java.util.Map;   public class Overhead {     @Test     public void useHashMap()     {         int i=0;         try {             Map<Integer, Payload> map = new HashMap<Integer, Payload>();             for (i=0; i < 4000000; i++) {                 int key = (int)(Math.random() * Integer.MAX_VALUE);                 map.put(key, new Payload(key));             }         }         catch (OutOfMemoryError e) {             System.out.println("Got up to: " + i);         }     }      @Test     public void useArrayList()     {         int i=0;         try {             ArrayListMap map = new ArrayListMap();             for (i=0; i < 9000000; i++) {                 int key = (int)(Math.random() * Integer.MAX_VALUE);                 map.put(key, new Payload(key));             }         }         catch (OutOfMemoryError e) {             System.out.println("Got up to: " + i);         }     } }   import java.util.ArrayList;   public class ArrayListMap {     private ArrayList<Payload> map = new ArrayList<Payload>();     private int[] primes = new int[128];      static boolean isPrime(int n)     {         for (int i=(int)Math.sqrt(n); i >= 2; i--) {             if (n % i == 0)                 return false;         }         return true;     }      ArrayListMap()     {         for (int i=0; i < 11000000; i++)    // this is clumsy, I admit             map.add(null);         int n=31;         for (int i=0; i < 128; i++) {             while (! isPrime(n))                 n+=2;             primes[i] = n;             n += 2;         }         System.out.println("Capacity = " + map.size());     }      public void put(int key, Payload value)     {         int hash = key % map.size();         int hash2 = primes[key % primes.length];         if (hash < 0)             hash += map.size();         do {             if (map.get(hash) == null) {                 map.set(hash, value);                 return;             }             hash += hash2;             if (hash >= map.size())                 hash -= map.size();         } while (true);     }      public Payload get(int key)     {         int hash = key % map.size();         int hash2 = primes[key % primes.length];         if (hash < 0)             hash += map.size();         do {             Payload payload = map.get(hash);             if (payload == null)                 return null;             if (payload.key == key)                 return payload;             hash += hash2;             if (hash >= map.size())                 hash -= map.size();         } while (true);     } } 
like image 128
Tim Cooper Avatar answered Oct 14 '22 12:10

Tim Cooper


The simplest thing would be to look at the source and work it out that way. However, you're really comparing apples and oranges - lists and maps are conceptually quite distinct. It's rare that you would choose between them on the basis of memory usage.

What's the background behind this question?

like image 28
Jon Skeet Avatar answered Oct 14 '22 10:10

Jon Skeet