Golang Map Internals

Publish date: 2025-03-18
Tags: Go, Interview-Questions

Key Points

Golang Map Internals


Map Creation and Structure

When you create a map using m := make(map[string]string), the Go runtime internally creates a map header structure. The variable m then points to this structure. The map header contains several critical components:

Each bucket in the map can store up to 8 key-value pairs. Additionally, each bucket includes an array of hash codes for faster comparisons during lookups, as well as an overflow pointer. The overflow pointer is used to link additional buckets if the current bucket becomes full.

Map Header
+-----------------------------+
| count       | int          |
| B           | int          |
| buckets_ptr | *BucketArray|
| hash_seed   | uint32      |
+-----------------------------+
        |
        v
Bucket Array
+---+---+---+---+
| B0 | B1 | B2 | ... |
+---+---+---+---+
        |
        v
Bucket Structure
+-----------------------------+
| hash_codes[8]               |
| key_value_pairs[8]         |
| overflow_ptr (*Bucket)      |
+-----------------------------+

Inserting New Values

When inserting a new key-value pair into a map, the following steps occur:

  1. Hash Computation: The Go runtime computes a hash code for the given key using a hash function. This computation incorporates a random hash seed to ensure uniqueness.
  2. Bucket Selection: Part of the hash code is used to determine which bucket should store the new key-value pair. Since the number of buckets is always a power of two, this step uses bit masking for efficient indexing.
  3. Collision Handling: The full hash code is compared with the hash codes already stored in the selected bucket. If no match is found, it means the key is new.
  4. Insertion:
    • If the bucket has an empty slot (i.e., it contains fewer than 8 key-value pairs), the new pair is added directly.
    • If the bucket is already full, a new overflow bucket is created, and the overflow pointer is updated to point to this new bucket. The new key-value pair is then stored in the overflow bucket.

This process ensures that the map can handle collisions effectively by chaining overflow buckets when necessary.

Initial State
+---+---+---+---+
| B0 | B1 | B2 | ... |
+---+---+---+---+
        |
        v
Bucket B0
+-----------------------------+
| hash_codes[8]               |
| key_value_pairs[7] + empty |
| overflow_ptr = NULL         |
+-----------------------------+

After Insertion (Overflow)
+---+---+---+---+
| B0 | B1 | B2 | ... |
+---+---+---+---+
        |
        v
Bucket B0
+-----------------------------+
| hash_codes[8]               |
| key_value_pairs[8]         |
| overflow_ptr -> OverflowBucket |
+-----------------------------+

Map Growth Mechanics

Go maps are designed to grow dynamically to maintain performance under varying conditions. Growth occurs under the following circumstances:

When growth occurs, the following actions take place:

This incremental growth strategy ensures that map operations remain responsive, even during resizing.

Before Growth
+---+---+---+---+
| B0 | B1 | B2 | ... |
+---+---+---+---+

After Growth
+---+---+---+---+---+---+---+---+
| B0 | B1 | B2 | ... | B3 | B4 | B5 | ... |
+---+---+---+---+---+---+---+---+
                |
                v
Old Bucket Array (for incremental copying)
+---+---+---+---+
| B0 | B1 | B2 | ... |
+---+---+---+---+

Deletion Behavior

Deleting an entry from a Go map involves removing it from its respective bucket and decrementing the count value in the map header. However, it is important to note that maps do not shrink. The number of buckets remains constant, even if all entries are deleted. This design choice means that maps only grow and never reduce their bucket count. As a result, engineers should be mindful of memory usage, especially in scenarios where maps undergo frequent deletions and insertions.


Performance Considerations


Comparative Insights

For software engineers, understanding the differences between maps and other data structures is essential. For example:


Practical Implications

Software engineers working with Go maps should consider the following:

By understanding these aspects, engineers can optimize Go applications, balancing performance, memory usage, and security.


Summary Table

Aspect Details
Internal Structure Maps are implemented as an array of buckets. Each bucket can store up to 8 key-value pairs.
Bucket Components Each bucket contains an array of hash codes, a list of key-value pairs, and an overflow pointer.
Map Header Contains metadata: number of entries, log2(buckets) (buckets are a power of two), pointer to bucket array, and a random hash seed.
Lookup Time O(1) due to hash functions.
Load Factor 6, triggers growth when bucket elements reach this limit (changed from 6.5 in Go 1.21).
Map Growth Doubles the number of buckets, redistributes old entries efficiently over time, maintains a pointer to the old bucket array.
Deletion Behavior Maps only grow; deleting all entries does not reduce the number of buckets.
Iteration Random; outputs differ each time the same map is printed.
Overflow Handling When a bucket is full (8 pairs) and receives a new value, a new bucket is created, and the overflow pointer points to it.
Tags: Go, Interview-Questions