Java Concurrency Tutorial – Thread-safe designs
After reviewing what the main risks are when dealing with concurrent programs (like atomicity or visibility), we will go through some class designs that will help us prevent the aforementioned bugs. Some of these designs result in the construction of thread-safe objects, allowing us to share them safely between threads. As an example, we will consider immutable and stateless objects. Other designs will prevent different threads from modifying the same data, like thread-local variables.
You can see all the source code at github.
1. Immutable objects
Immutable objects have a state (have data which represent the object’s state), but it is built upon construction, and once the object is instantiated, the state cannot be modified.
Although threads may interleave, the object has only one possible state. Since all fields are read-only, not a single thread will be able to change object’s data. For this reason, an immutable object is inherently thread-safe.
Product shows an example of an immutable class. It builds all its data during construction and none of its fields are modifiable:
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 | public final class Product { private final String id; private final String name; private final double price; public Product(String id, String name, double price) { this .id = id; this .name = name; this .price = price; } public String getId() { return this .id; } public String getName() { return this .name; } public double getPrice() { return this .price; } public String toString() { return new StringBuilder( this .id).append( "-" ).append( this .name) .append( " (" ).append( this .price).append( ")" ).toString(); } public boolean equals(Object x) { if ( this == x) return true ; if (x == null ) return false ; if ( this .getClass() != x.getClass()) return false ; Product that = (Product) x; if (! this .id.equals(that.id)) return false ; if (! this .name.equals(that.name)) return false ; if ( this .price != that.price) return false ; return true ; } public int hashCode() { int hash = 17 ; hash = 31 * hash + this .getId().hashCode(); hash = 31 * hash + this .getName().hashCode(); hash = 31 * hash + ((Double) this .getPrice()).hashCode(); return hash; } } |
In some cases, it won’t be sufficient to make a field final. For example, MutableProduct class is not immutable although all fields are final:
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 | public final class MutableProduct { private final String id; private final String name; private final double price; private final List<String> categories = new ArrayList<>(); public MutableProduct(String id, String name, double price) { this .id = id; this .name = name; this .price = price; this .categories.add( "A" ); this .categories.add( "B" ); this .categories.add( "C" ); } public String getId() { return this .id; } public String getName() { return this .name; } public double getPrice() { return this .price; } public List<String> getCategories() { return this .categories; } public List<String> getCategoriesUnmodifiable() { return Collections.unmodifiableList(categories); } public String toString() { return new StringBuilder( this .id).append( "-" ).append( this .name) .append( " (" ).append( this .price).append( ")" ).toString(); } } |
Why is the above class not immutable? The reason is we let a reference to escape from the scope of its class. The field ‘categories‘ is a mutable reference, so after returning it, the client could modify it. In order to show this, consider the following program:
01 02 03 04 05 06 07 08 09 10 | public static void main(String[] args) { MutableProduct p = new MutableProduct( "1" , "a product" , 43.00 ); System.out.println( "Product categories" ); for (String c : p.getCategories()) System.out.println(c); p.getCategories().remove( 0 ); System.out.println( "\nModified Product categories" ); for (String c : p.getCategories()) System.out.println(c); } |
And the console output:
1 2 3 4 5 6 7 | Product categories A B C |
1 2 3 4 5 | Modified Product categories B C |
Since categories field is mutable and it escaped the object’s scope, the client has modified the categories list. The product, which was supposed to be immutable, has been modified, leading to a new state.
If you want to expose the content of the list, you could use an unmodifiable view of the list:
1 2 3 | public List<String> getCategoriesUnmodifiable() { return Collections.unmodifiableList(categories); } |
2. Stateless objects
Stateless objects are similar to immutable objects but in this case, they do not have a state, not even one. When an object is stateless it does not have to remember any data between invocations.
Since there is no state to modify, one thread will not be able to affect the result of another thread invoking the object’s operations. For this reason, a stateless class is inherently thread-safe.
ProductHandler is an example of this type of objects. It contains several operations over Product objects and it does not store any data between invocations. The result of an operation does not depend on previous invocations or any stored data:
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 | public class ProductHandler { private static final int DISCOUNT = 90 ; public Product applyDiscount(Product p) { double finalPrice = p.getPrice() * DISCOUNT / 100 ; return new Product(p.getId(), p.getName(), finalPrice); } public double sumCart(List<Product> cart) { double total = 0.0 ; for (Product p : cart.toArray( new Product[ 0 ])) total += p.getPrice(); return total; } } |
In its sumCart method, the ProductHandler converts the product list to an array since for-each loop uses an iterator internally to iterate through its elements. List iterators are not thread-safe and could throw a ConcurrentModificationException if modified during iteration. Depending on your needs, you might choose a different strategy.
3. Thread-local variables
Thread-local variables are those variables defined within the scope of a thread. No other threads will see nor modify them.
The first type is local variables. In the below example, the total variable is stored in the thread’s stack:
1 2 3 4 5 6 | public double sumCart(List<Product> cart) { double total = 0.0 ; for (Product p : cart.toArray( new Product[ 0 ])) total += p.getPrice(); return total; } |
Just take into account that if instead of a primitive you define a reference and return it, it will escape its scope. You may not know where the returned reference is stored. The code that calls sumCartmethod could store it in a static field and allow it being shared between different threads.
The second type is ThreadLocal class. This class provides a storage independent for each thread. Values stored into an instance of ThreadLocal are accessible from any code within the same thread.
The ClientRequestId class shows an example of ThreadLocal usage:
01 02 03 04 05 06 07 08 09 10 11 12 | public class ClientRequestId { private static final ThreadLocal<String> id = new ThreadLocal<String>() { @Override protected String initialValue() { return UUID.randomUUID().toString(); } }; public static String get() { return id.get(); } } |
The ProductHandlerThreadLocal class uses ClientRequestId to return the same generated id within the same thread:
1 2 3 4 5 6 7 | public class ProductHandlerThreadLocal { //Same methods as in ProductHandler class public String generateOrderId() { return ClientRequestId.get(); } } |
If you execute the main method, the console output will show different ids for each thread. As an example:
1 2 3 4 5 6 7 8 9 | T1 - 23dccaa2-8f34-43ec-bbfa-01cec5df3258 T2 - 936d0d9d-b507-46c0-a264-4b51ac3f527d T2 - 936d0d9d-b507-46c0-a264-4b51ac3f527d T3 - 126b8359-3bcc-46b9-859a-d305aff22c7e ... |
If you are going to use ThreadLocal, you should care about some of the risks of using it when threads are pooled (like in application servers). You could end up with memory leaks or information leaking between requests. I won’t extend myself in this subject since the post How to shoot yourself in foot with ThreadLocals explains well how this can happen.
4. Using synchronization
Another way of providing thread-safe access to objects is through synchronization. If we synchronize all accesses to a reference, only a single thread will access it at a given time. We will discuss this on further posts.
5. Conclusion
We have seen several techniques that help us build simpler objects that can be shared safely between threads. It is much harder to prevent concurrent bugs if an object can have multiple states. On the other hand, if an object can have only one state or none, we won’t have to worry about different threads accessing it at the same time.
Reference: | Java Concurrency Tutorial – Thread-safe designs from our JCG partner Xavier Padro at the Xavier Padró’s Blog blog. |
Thanks for the post. It is nice review basics of threads.