Clojure Atoms

Race condition free concurrency in Clojure.

Mark Nemec
3 min readDec 30, 2014

For my honours project I need to program in Clojure, a dialect of Lisp that targets the Java virtual machine. I wrote this blog post mainly because writing about something forces me to think about it more and understand it better.

Clojure data structures are immutable. This is so that programs written in Clojure are more functional and hence more robust. To provide mutable state, Clojure provides four reference types: vars, refs, agents, and atoms. Every type is useful in a different scenario but this post focuses on atoms.

Definition

The Clojure.org website describes atoms as follows:

Atoms provide a way to manage shared, synchronous, independent state.

Let’s break this statement down.

Shared state

In this context, shared state means sharing some state between two or more concurrent processes. For example, we could have two threads accessing a value in the same memory location, one trying to update the value and the other trying to read the value.

In most languages, we would need to take care to avoid race conditions. With atoms, however, this is handled for the programmer automatically using the compare-and-swap instruction. Thus, changes to atoms are always free of race conditions.

Synchronous state

Atoms operate in a synchronous manner. That is if we de-reference an atom, we get its value immediately as opposed to some time in the future when the value becomes “known”. Moreover, if we try to update the value of an atom with an operation that takes e.g. 5 seconds to complete, the thread will block until the operation is completed.

Independent state

Even though atoms are shared by multiple threads, the state they provide is independent. This means that changing the value of an atom does not and should not affect some other state.

A typical scenario where this is not true is moving money between bank accounts. If a bank needs to move some money from account A to account B it either wants to transfer all of it or none of it. It doesn’t want to leave the accounts in some inconsistent state, for example withdrawing money from A without ever depositing it in B. In this scenario we would want to use refs which are created specifically for coordinated access.

A good example of where we could use an atom is a counter.

Creating an atom

Creating an atom is easy, we simply use the atom function:

user=> (def counter (atom 0))#’user/counteruser=> counter#<Atom@1cdca0c8: 0>

Here we defined an atom called counter in namespace user. Its value is 0.

Reading an atom

To read the value of an atom we use the @ reader macro or the deref function:

user=> @counter0user=> (deref counter)0

Updating an atom

To update the value associated with the atom we use the swap! function:

user=> @counter0user=> (swap! counter inc)1user=> @counter1

Function swap! is defined as:

(swap! atom f args)

It atomically swaps the value of the atom to be:

(apply inc current-value-of-atom args)

In our case args is empty and thus we got 1 as a result:

user=> (apply inc @counter [])1

Since the value of the atom could have been changed by some other thread before it was swapped by the current thread swap! may need to be retried until successful and thus the function that it applies has to be free of side effects. In our case this is true since 0 + 1 is always equal to 1.

As you can see, atoms can be very powerful even though working with them is quite simple. They are also fast:

Sources

--

--