Skip to content
forked from cornelk/hashmap

A Golang lock-free thread-safe HashMap optimized for fastest read access.

License

Notifications You must be signed in to change notification settings

tigrato/hashmap

Repository files navigation

hashmap Build Status GoDoc Go Report Card codecov

Overview

A Golang lock-free thread-safe HashMap optimized for fastest read access.

Usage

Set a value for a key in the map:

m := &HashMap{}
m.Set("amount", 123)

Read a value for a key from the map:

amount, ok := m.Get("amount")

Use the map to count URL requests:

var i int64
actual, _ := m.GetOrInsert("api/123", &i)
counter := (actual).(*int64)
atomic.AddInt64(counter, 1) // increase counter
...
count := atomic.LoadInt64(counter) // read counter

Benchmarks

Reading from the hash map in a thread-safe way is nearly as fast as reading from a standard Golang map in an unsafe way and twice as fast as Go's sync.Map:

BenchmarkReadHashMapUint-8                	  200000	      6841 ns/op
BenchmarkReadGoMapUintUnsafe-8            	  300000	      6061 ns/op
BenchmarkReadGoMapUintMutex-8             	   30000	     40283 ns/op
BenchmarkReadGoSyncMapUint-8              	  100000	     14657 ns/op

If your keys for the map are already hashes, no extra hashing needs to be done by the map:

BenchmarkReadHashMapHashedKey-8           	 1000000	      1834 ns/op

Reading from the map while writes are happening:

BenchmarkReadHashMapWithWritesUint-8      	  200000	      9282 ns/op
BenchmarkReadGoMapWithWritesUintMutex-8   	   10000	    121994 ns/op
BenchmarkReadGoSyncMapWithWritesUint-8    	  100000	     15434 ns/op

Write performance without any concurrent reads:

BenchmarkWriteHashMapUint-8               	   10000	    227535 ns/op
BenchmarkWriteGoMapMutexUint-8            	   30000	     52072 ns/op
BenchmarkWriteGoSyncMapUint-8             	   10000	    181075 ns/op

The benchmarks were run with Golang 1.10.1 on MacOS.

Benefits over Golangs builtin map

  • Faster

  • thread-safe access without need of a(n extra) mutex

  • Compare-and-swap access for values

  • Access functions for keys that are hashes and do not need to be hashed again

Benefits over Golangs sync.Map

  • Faster

  • Access functions for keys that are hashes and do not need to be hashed again

Technical details

  • Technical design decisions have been made based on benchmarks that are stored in an external repository: go-benchmark

  • The library uses a sorted doubly linked list and a slice as an index into that list.

  • The Get() function contains helper functions that have been inlined manually until the Golang compiler will inline them automatically.

  • It optimizes the slice access by circumventing the Golang size check when reading from the slice. Once a slice is allocated, the size of it does not change. The library limits the index into the slice, therefor the Golang size check is obsolete. When the slice reaches a defined fill rate, a bigger slice is allocated and all keys are recalculated and transferred into the new slice.

About

A Golang lock-free thread-safe HashMap optimized for fastest read access.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Go 100.0%