You can find all the code for this chapter here
In arrays & slices, you saw how to store values in order. Now, we will
look at a way to store items by a key
and look them up quickly.
Maps allow you to store items in a manner similar to a dictionary. You can
think of the key
as the word and the value
as the definition. And what better
way is there to learn about Maps than to build our own dictionary?
In dictionary_test.go
package main
import "testing"
func TestSearch(t *testing.T) {
dictionary := map[string]string{"test": "this is just a test"}
got := Search(dictionary, "test")
want := "this is just a test"
if got != want {
t.Errorf("got '%s' want '%s' given, '%s'", got, want, "test")
}
}
Declaring a Map is somewhat similar to an array. Except, it starts with the
map
keyword and requires two types. The first is the key, which is written
inside the []
. The second is the value, which goes right after the []
.
The key is special. It can only be a comparable type because without the ability to tell if 2 keys are equal, we have no way to ensure that we are getting the correct value. Comparable types are explained in depth in the language spec.
The value, on the other hand, can be any type you want. It can even be another Map.
Everything else in this test should be familiar.
By running go test
the compiler will fail with ./dictionary_test.go:8:9: undefined: Search
.
In dictionary.go
package main
func Search(dictionary map[string]string, word string) string {
return ""
}
Your test should now fail with a clear error message
dictionary_test.go:12: got '' want 'this is just a test' given, 'test'
.
func Search(dictionary map[string]string, word string) string {
return dictionary[word]
}
Getting a value out of a Map is the same as getting a value out of Array
map[key]
.
func TestSearch(t *testing.T) {
dictionary := map[string]string{"test": "this is just a test"}
got := Search(dictionary, "test")
want := "this is just a test"
assertStrings(t, got, want)
}
func assertStrings(t *testing.T, got, want string) {
t.Helper()
if got != want {
t.Errorf("got '%s' want '%s'", got, want)
}
}
I decided to create an assertStrings
helper and get rid of the
given
piece to make the implementation more general.
We can improve our dictionary's usage by creating a new type around Map
and making Search
a method.
In dictionary_test.go
:
func TestSearch(t *testing.T) {
dictionary := Dictionary{"test": "this is just a test"}
got := dictionary.Search("test")
want := "this is just a test"
assertStrings(t, got, want)
}
We started using the Dictionary
type, which we have not defined yet. Then called
Search
on the Dictionary
instance.
We did not need to change assertStrings
.
In dictionary.go
:
type Dictionary map[string]string
func (d Dictionary) Search(word string) string {
return d[word]
}
Here we created a Dictionary
type which acts as a thin wrapper around
map
. With the custom type defined, we can create the Search
method.
The basic search was very easy to implement, but what will happen if we supply a word that's not in our dictionary?
We actually get nothing back. This is good because the program can continue to run, but there is a better approach. The function can report that the word is not in the dictionary. This way, the user isn't left wondering if the word doesn't exist or if there is just no definition (this might not seem very useful for a dictionary. However, it's a scenario that could be key in other usecases).
func TestSearch(t *testing.T) {
dictionary := Dictionary{"test": "this is just a test"}
t.Run("known word", func(t *testing.T) {
got, _ := dictionary.Search("test")
want := "this is just a test"
assertStrings(t, got, want)
})
t.Run("unknown word", func(t *testing.T) {
_, err := dictionary.Search("unknown")
want := "could not find the word you were looking for"
if err == nil {
t.Fatal("expected to get an error.")
}
assertStrings(t, err.Error(), want)
})
}
The way to handle this scenario in Go is to return a second argument which is
an Error
type.
Error
s can be converted to a string with the .Error()
method, which we do when passing it to the assertion. We are also protecting
assertStrings
with if
to ensure we don't call .Error()
on nil
.
This does not compile
./dictionary_test.go:18:10: assignment mismatch: 2 variables but 1 values
func (d Dictionary) Search(word string) (string, error) {
return d[word], nil
}
Your test should now fail with a much clearer error message.
dictionary_test.go:22: expected to get an error.
func (d Dictionary) Search(word string) (string, error) {
definition, ok := d[word]
if !ok {
return "", errors.New("could not find the word you were looking for")
}
return definition, nil
}
In order to make this pass we are using an interesting property of the Map lookup. It can return 2 values. The second value being a boolean which indicates if the key was found successfully.
This property allows us to differentiate between a word that doesn't exist and a word that just doesn't have a definition.
var ErrNotFound = errors.New("could not find the word you were looking for")
func (d Dictionary) Search(word string) (string, error) {
definition, ok := d[word]
if !ok {
return "", ErrNotFound
}
return definition, nil
}
We can get rid of the magic error in our Search
function by extracting it
into a variable. This will also allow us to have a better test.
t.Run("unknown word", func(t *testing.T) {
_, got := dictionary.Search("unknown")
assertError(t, got, ErrNotFound)
})
func assertError(t *testing.T, got, want error) {
t.Helper()
if got != want {
t.Errorf("got error '%s' want '%s'", got, want)
}
}
By creating a new helper we were able to simplify our test, and start using
our ErrNotFound
variable so our test doesn't fail if we change the error
text in the future.
We have a great way to search the dictionary. However, we have no way to add new words to our dictionary.
func TestAdd(t *testing.T) {
dictionary := Dictionary{}
dictionary.Add("test", "this is just a test")
want := "this is just a test"
got, err := dictionary.Search("test")
if err != nil {
t.Fatal("should find added word:", err)
}
if want != got {
t.Errorf("got '%s' want '%s'", got, want)
}
}
In this test, we are utilizing our Search
function to make the
validation of the dictionary a little easier.
In dictionary.go
func (d Dictionary) Add(word, definition string) {
}
Your test should now fail
dictionary_test.go:31: should find added word: could not find the word you were
looking for
func (d Dictionary) Add(word, definition string) {
d[word] = definition
}
Adding to a Map is also similar to an Array. You just need to specify key and set it equal to a value.
An interesting property of Maps is that you can modify them without passing
them as a pointer. This is because map
is a reference type. Meaning it holds
a reference to the underlying data structure, much like a pointer. The
underlying data structure is a hash table
, or hash map
, and you can read
more about hash tables
here.
Maps being a reference is really good, because no matter how big a Map gets there will only be one copy.
A gotcha that reference types introduce is that maps can be a nil
value. If
you try to use a nil
Map, you will get a nil pointer exception
which will
stop your program's execution in it's tracks.
Because of nil pointer exceptions
, you should never initialize an empty Map
variable:
var m map[string]string
Instead you can initialize an empty map like we were doing above, or use the
make
keyword to create a map for you:
dictionary = map[string]string{}
// OR
dictionary = make(map[string]string)
Both approaches create an empty hash map
and point dictionary
at it. Which
ensures that you will never get a nil pointer exception
.
There isn't much to refactor in our implementation but the test could use a little simplification.
func TestAdd(t *testing.T) {
dictionary := Dictionary{}
word := "test"
definition := "this is just a test"
dictionary.Add(word, definition)
assertDefinition(t, dictionary, word, definition)
}
func assertDefinition(t *testing.T, dictionary Dictionary, word, definition string) {
t.Helper()
got, err := dictionary.Search(word)
if err != nil {
t.Fatal("should find added word:", err)
}
if definition != got {
t.Errorf("got '%s' want '%s'", got, definition)
}
}
We made variables for word and definition, and moved the definition assertion into its own helper function.
Our Add
is looking good. Except, we didn't consider what happens when the
value we are trying to add already exists!
Map will not throw an error if the value already exists. Instead, they will go
ahead and overwrite the value with the newly provided value. This can
be convenient in practice, but makes our function name less than
accurate. Add
should not modify existing values. It should only add new
words to our dictionary.
func TestAdd(t *testing.T) {
t.Run("new word", func(t *testing.T) {
dictionary := Dictionary{}
word := "test"
definition := "this is just a test"
err := dictionary.Add(word, definition)
assertError(t, err, nil)
assertDefinition(t, dictionary, word, definition)
})
t.Run("existing word", func(t *testing.T) {
word := "test"
definition := "this is just a test"
dictionary := Dictionary{word: definition}
err := dictionary.Add(word, "new test")
assertError(t, err, ErrWordExists)
assertDefinition(t, dictionary, word, definition)
})
}
For this test we modified Add
to return an error, which we are
validating against a new error variable, ErrWordExists
. We also modified
the previous test to check for a nil
error.
The compiler will fail because we are not returning a value for Add
.
./dictionary_test.go:30:13: dictionary.Add(word, definition) used as value
./dictionary_test.go:41:13: dictionary.Add(word, "new test") used as value
In dictionary.go
var (
ErrNotFound = errors.New("could not find the word you were looking for")
ErrWordExists = errors.New("cannot add word because it already exists")
)
func (d Dictionary) Add(word, definition string) error {
d[word] = definition
return nil
}
Now we get two more errors. We are still modifying the value, and
returning a nil
error.
dictionary_test.go:43: got error '%!s(<nil>)' want 'cannot add word because
it already exists'
dictionary_test.go:44: got 'new test' want 'this is just a test'
func (d Dictionary) Add(word, definition string) error {
_, err := d.Search(word)
switch err {
case ErrNotFound:
d[word] = definition
case nil:
return ErrWordExists
default:
return err
}
return nil
}
Here we are using a switch
statement to match on the error. Having a switch
like this provides an extra safety net, in case Search
returns an error other than ErrNotFound
.
We don't have too much to refactor, but as our error usage grows we can make a few modifications.
const (
ErrNotFound = DictionaryErr("could not find the word you were looking for")
ErrWordExists = DictionaryErr("cannot add word because it already exists")
)
type DictionaryErr string
func (e DictionaryErr) Error() string {
return string(e)
}
We made the errors constant; this required us to create our own DictionaryErr
type which implements the error
interface. You can read more about the details in this excellent article by Dave
Cheney. Simply put, it makes the errors more reusable and immutable.
func TestUpdate(t *testing.T) {
word := "test"
definition := "this is just a test"
dictionary := Dictionary{word: definition}
newDefinition := "new definition"
dictionary.Update(word, newDefinition)
assertDefinition(t, dictionary, word, newDefinition)
}
Update
is very closely related to Create
and will be our next
implementation.
./dictionary_test.go:53:2: dictionary.Update undefined (type Dictionary has no field or method Update)
We already know how to deal with an error like this. We need to define our function.
func (d Dictionary) Update(word, definition string) {}
With that in place we are able to see that we need to change the definition of the word.
dictionary_test.go:55: got 'this is just a test' want 'new definition'
We already saw how to do this when we fixed the issue with create. So let's implement something really similar to create.
func (d Dictionary) Update(word, definition string) {
d[word] = definition
}
There is no refactoring we need to do on this since it was a simple change.
However, we now have the same issue as with create. If we pass in a new word,
Update
will add it to the dictionary.
t.Run("existing word", func(t *testing.T) {
word := "test"
definition := "this is just a test"
newDefinition := "new definition"
dictionary := Dictionary{word: definition}
err := dictionary.Update(word, newDefinition)
assertError(t, err, nil)
assertDefinition(t, dictionary, word, newDefinition)
})
t.Run("new word", func(t *testing.T) {
word := "test"
definition := "this is just a test"
dictionary := Dictionary{}
err := dictionary.Update(dictionaryword, definition)
assertError(t, err, ErrWordDoesNotExist)
})
We added yet another error type for when the word does not exist. We also
modified Update
to return an error
value.
./dictionary_test.go:53:16: dictionary.Update(word, "new test") used as value
./dictionary_test.go:64:16: dictionary.Update(word, definition) used as value
./dictionary_test.go:66:23: undefined: ErrWordDoesNotExists
We get 3 errors this time, but we know how to deal with these.
const (
ErrNotFound = DictionaryErr("could not find the word you were looking for")
ErrWordExists = DictionaryErr("cannot add word because it already exists")
ErrWordDoesNotExist = DictionaryErr("cannot update word because it does not exist")
)
func (d Dictionary) Update(word, definition string) error {
d[word] = definition
return nil
}
We added our own error type and are returning a nil
error.
With these changes, we now get a very clear error:
dictionary_test.go:66: got error '%!s(<nil>)' want 'cannot update word because it does not exist'
func (d Dictionary) Update(word, definition string) error {
_, err := dictionary.Search(word)
switch err {
case ErrNotFound:
return ErrWordDoesNotExist
case nil:
d[word] = definition
default:
return err
}
return nil
}
This function looks almost identical to Add
except we switched when we update
the dictionary
and when we return an error.
We could reuse ErrNotFound
and not add a new error. However, it is often
better to have a precise error for when an update fails.
Having specific errors gives you more information about what went wrong. Here is an example in a web app:
You can redirect the user when
ErrNotFound
is encountered, but display an error message whenErrWordDoesNotExist
is encountered.
func TestDelete(t *testing.T) {
word := "test"
dictionary := Dictionary{word: "test definition"}
dictionary.Delete(word)
_, err := dictionary.Search(word)
if err != ErrNotFound {
t.Errorf("Expected '%s' to be deleted", word)
}
}
Our test creates a Dictionary
with a word and then checks if the word has been
removed.
By running go test
we get:
./dictionary_test.go:74:6: dictionary.Delete undefined (type Dictionary has no field or method Delete)
func (d Dictionary) Delete(word string) {
}
After we add this, the test tells us we are not deleting the word.
dictionary_test.go:78: Expected 'test' to be deleted
func (d Dictionary) Delete(word string) {
delete(d, word)
}
Go has a built in function delete
that works on maps. It takes two arguments.
The first is the map and the second is the key to be removed.
The delete
function returns nothing, and we based our Delete
method on
the same notion. Since deleting a value that's not there has no effect, unlike
our Update
and `Create** methods, we don't need to complicate the API with errors.
In this section we covered a lot. We made a full CRUD API for our dictionary. Throughout the process we learned how to:
- Create maps
- Search for items in maps
- Add new items to maps
- Update items in maps
- Delete items from a map
- Learned more about errors
- How to create errors that are constants
- Writing error wrappers