Immutability in Java (2 Part Series)
1 Immutability in Java Made Easy
2 ️ Immutables/AutoValue/Lombok Which One?
The article was initially published at carloschac.in
See also:
️ Immutables/AutoValue/Lombok Which One?
Carlos Chacin ・ Apr 13 ’20
#java #bestpractices #immutability
In Effective Java, Joshua Bloch makes the following recommendation:
Classes should be immutable unless there’s a very good reason to make them mutable… If a class cannot be made immutable, limit its mutability as much as possible.
Immutable Objects
An object is considered immutable if its state cannot change after it is constructed. Maximum reliance on immutable objects is widely accepted as a sound strategy for creating a simple, reliable code. reference
- Immutable objects are constructed once, in a consistent state, and can be safely shared
- Will fail if mandatory attributes are missing
- Cannot be sneakily modified when passed to other code
- Immutable objects are naturally thread-safe and can therefore be safely shared among threads
- No excessive copying
- No excessive synchronization
- Object definitions are pleasant to write and read
- No boilerplate setter and getters
- No ugly IDE-generated
hashCode
,equals
andtoString
methods that end up being stored in source control. reference
Let’s convert a mutable object into an immutable one (by hand ):
The following class is what we usually call POJO or Java Bean:
import java.util.Date;
import java.util.List;
import java.util.Objects;
public class OldModel {
private String fieldA;
private Date fieldB;
private Long fieldC;
private List<String> fieldD;
public OldModel() {
}
public String getFieldA() {
return fieldA;
}
public void setFieldA(String fieldA) {
this.fieldA = fieldA;
}
public Date getFieldB() {
return fieldB;
}
public void setFieldB(Date fieldB) {
this.fieldB = fieldB;
}
public Long getFieldC() {
return fieldC;
}
public void setFieldC(Long fieldC) {
this.fieldC = fieldC;
}
public List<String> getFieldD() {
return fieldD;
}
public void setFieldD(List<String> fieldD) {
this.fieldD = fieldD;
}
@Override
public String toString() {
return "OldModel{" +
"fieldA='" + fieldA + '\'' +
", fieldB=" + fieldB +
", fieldC=" + fieldC +
", fieldD=" + fieldD +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
OldModel oldModel = (OldModel) o;
return Objects.equals(getFieldA(), oldModel.getFieldA()) &&
Objects.equals(getFieldB(), oldModel.getFieldB()) &&
Objects.equals(getFieldC(), oldModel.getFieldC()) &&
Objects.equals(getFieldD(), oldModel.getFieldD());
}
@Override
public int hashCode() {
return Objects.hash(getFieldA(), getFieldB(), getFieldC(), getFieldD());
}
}
Enter fullscreen mode Exit fullscreen mode
To convert that to an immutable object, we have to:
The class can not be overridden:
- Make the class final.
- Or make the constructor final and use static factory methods.
-public class OldModel { +public final class OldModel {
Enter fullscreen mode Exit fullscreen mode
Make all the fields private and final
- private String fieldA; - private Date fieldB; - private Long fieldC; - private List<String> fieldD; + private final String fieldA; + private final Date fieldB; + private final Long fieldC; + private final List<String> fieldD;
Enter fullscreen mode Exit fullscreen mode
The object has to be constructed in 1️⃣ single step.
- public OldModel() { + public OldModel( + final String fieldA, + final Date fieldB, + final Long fieldC, + final List<String> fieldD) { + this.fieldA = fieldA; + this.fieldB = fieldB; + this.fieldC = fieldC; + this.fieldD = fieldD; }
Enter fullscreen mode Exit fullscreen mode
Do not provide methods that can change the object state.
- public void setFieldA(String fieldA) { - this.fieldA = fieldA; - } .
. // Remove all the setters
.
Enter fullscreen mode Exit fullscreen mode
⏩ If any of the fields is a mutable object, provide a defensive copy of that object instead.
+ public Date getFieldB() { + return new Date(fieldB.getTime()); // Easy to forget about this :( + } + + public List<String> getFieldD() { + return Collections.unmodifiableList(fieldD); // This is not great :( + }
Enter fullscreen mode Exit fullscreen mode
About manually doing this
Well, this is much work, and it is error-prone as well, and even when we can make the IDE vomit all that code for us, we still need to check and modify certain things.
And this is the result:
import java.util.Collections;
import java.util.Date;
import java.util.List;
import java.util.Objects;
public final class OldModel {
private final String fieldA;
private final Date fieldB;
private final Long fieldC;
private final List<String> fieldD;
public OldModel(
final String fieldA,
final Date fieldB,
final Long fieldC,
final List<String> fieldD) {
this.fieldA = fieldA;
this.fieldB = fieldB;
this.fieldC = fieldC;
this.fieldD = fieldD;
}
public String getFieldA() {
return this.fieldA;
}
public Date getFieldB() {
return new Date(this.fieldB.getTime()); // Easy to forget about this :(
}
public Long getFieldC() {
return this.fieldC;
}
public List<String> getFieldD() {
return Collections.unmodifiableList(this.fieldD); // This is not great :(
}
// toSting, equals and hashCode omitted
}
Enter fullscreen mode Exit fullscreen mode
Let’s do it now using the easy way
Immutables Library
Java annotation processors to generate simple, safe and consistent value objects. Do not repeat yourself, try Immutables, the most comprehensive tool in this field!
Include the immutables.org dependency to your project:
Maven Dependency:
<dependency>
<groupId>org.immutables</groupId>
<artifactId>value</artifactId>
<version>2.8.3</version>
<scope>provided</scope>
</dependency>
Enter fullscreen mode Exit fullscreen mode
Create the immutable object
import org.immutables.value.Value;
import java.util.Date;
import java.util.List;
@Value.Immutable
public interface NewModel {
String fieldA();
Date fieldB();
Long fieldC();
List<String> fieldD();
}
Enter fullscreen mode Exit fullscreen mode
Compile and Enjoy
After compilation, the annotation processor would generate the following code for you:
A generated
final
class that extends a manually-written interface value type and implements all declared accessor methods as well as supporting fields, methods, constructors, and a builder class.An immutable implementation class implements abstract attribute accessors for scalar primitive and object reference types, with special support provided for collection attributes and other types. java.lang.Object’s
equals
,hashCode
, andtoString
methods are overridden and fully dependent on attribute values rather than on object identity.Immutable implementation classes are the primary (but not the only) source code artifacts generated by the Immutables annotation processor.
️ How to use the generated immutable class
public class App {
public static void main(String[] args) {
final ImmutableNewModel model = ImmutableNewModel.builder()
.fieldA("A")
.fieldB(new Date())
.fieldC(1L)
.addFieldD("a", "b", "d")
.addFieldD("e")
.build();
System.out.println(model);
}
}
Enter fullscreen mode Exit fullscreen mode
Output:
NewModel{fieldA=A, fieldB=Sat Apr 11 15:53:26 PDT 2020, fieldC=1, fieldD=[a, b, d, e]}
Enter fullscreen mode Exit fullscreen mode
A few comparisons
OldModel | NewModel | |
---|---|---|
Lines of code to maintain | 69 | 16 |
Lines of code to generated | 0 | 377 |
Defensive copy of fields | ⁉️ | |
Fluent API for copy | ||
Builder | ||
Nullability Checks |
The generated code
By default, on maven projects, the compiler would generate and auto import the generated code to and from target/generated_sources
folder, notice that most of the time, we ignore the target/
folder in version control systems (VCS) like git and mercurial. using a .gitinore
file in the project. This code should not have to be pushed to the centralized VCS.
️️ WARNING: It’s a lot of code ️️
package com.groupon.api.talks;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.List;
import java.util.Objects;
import org.immutables.value.Generated;
/** * Immutable implementation of {@link NewModel}. * <p> * Use the builder to create immutable instances: * {@code ImmutableNewModel.builder()}. */
@Generated(from = "NewModel", generator = "Immutables")
@SuppressWarnings({"all"})
@javax.annotation.Generated("org.immutables.processor.ProxyProcessor")
public final class ImmutableNewModel implements NewModel {
private final String fieldA;
private final Date fieldB;
private final Long fieldC;
private final List<String> fieldD;
private ImmutableNewModel(
String fieldA,
Date fieldB,
Long fieldC,
List<String> fieldD) {
this.fieldA = fieldA;
this.fieldB = fieldB;
this.fieldC = fieldC;
this.fieldD = fieldD;
}
/** * @return The value of the {@code fieldA} attribute */
@Override
public String fieldA() {
return fieldA;
}
/** * @return The value of the {@code fieldB} attribute */
@Override
public Date fieldB() {
return fieldB;
}
/** * @return The value of the {@code fieldC} attribute */
@Override
public Long fieldC() {
return fieldC;
}
/** * @return The value of the {@code fieldD} attribute */
@Override
public List<String> fieldD() {
return fieldD;
}
/** * Copy the current immutable object by setting a value for the {@link NewModel#fieldA() fieldA} attribute. * An equals check used to prevent copying of the same value by returning {@code this}. * @param value A new value for fieldA * @return A modified copy of the {@code this} object */
public final ImmutableNewModel withFieldA(String value) {
String newValue = Objects.requireNonNull(value, "fieldA");
if (this.fieldA.equals(newValue)) return this;
return new ImmutableNewModel(newValue, this.fieldB, this.fieldC, this.fieldD);
}
/** * Copy the current immutable object by setting a value for the {@link NewModel#fieldB() fieldB} attribute. * A shallow reference equality check is used to prevent copying of the same value by returning {@code this}. * @param value A new value for fieldB * @return A modified copy of the {@code this} object */
public final ImmutableNewModel withFieldB(Date value) {
if (this.fieldB == value) return this;
Date newValue = Objects.requireNonNull(value, "fieldB");
return new ImmutableNewModel(this.fieldA, newValue, this.fieldC, this.fieldD);
}
/** * Copy the current immutable object by setting a value for the {@link NewModel#fieldC() fieldC} attribute. * An equals check used to prevent copying of the same value by returning {@code this}. * @param value A new value for fieldC * @return A modified copy of the {@code this} object */
public final ImmutableNewModel withFieldC(Long value) {
Long newValue = Objects.requireNonNull(value, "fieldC");
if (this.fieldC.equals(newValue)) return this;
return new ImmutableNewModel(this.fieldA, this.fieldB, newValue, this.fieldD);
}
/** * Copy the current immutable object with elements that replace the content of {@link NewModel#fieldD() fieldD}. * @param elements The elements to set * @return A modified copy of {@code this} object */
public final ImmutableNewModel withFieldD(String... elements) {
List<String> newValue = createUnmodifiableList(false, createSafeList(Arrays.asList(elements), true, false));
return new ImmutableNewModel(this.fieldA, this.fieldB, this.fieldC, newValue);
}
/** * Copy the current immutable object with elements that replace the content of {@link NewModel#fieldD() fieldD}. * A shallow reference equality check is used to prevent copying of the same value by returning {@code this}. * @param elements An iterable of fieldD elements to set * @return A modified copy of {@code this} object */
public final ImmutableNewModel withFieldD(Iterable<String> elements) {
if (this.fieldD == elements) return this;
List<String> newValue = createUnmodifiableList(false, createSafeList(elements, true, false));
return new ImmutableNewModel(this.fieldA, this.fieldB, this.fieldC, newValue);
}
/** * This instance is equal to all instances of {@code ImmutableNewModel} that have equal attribute values. * @return {@code true} if {@code this} is equal to {@code another} instance */
@Override
public boolean equals(Object another) {
if (this == another) return true;
return another instanceof ImmutableNewModel
&& equalTo((ImmutableNewModel) another);
}
private boolean equalTo(ImmutableNewModel another) {
return fieldA.equals(another.fieldA)
&& fieldB.equals(another.fieldB)
&& fieldC.equals(another.fieldC)
&& fieldD.equals(another.fieldD);
}
/** * Computes a hash code from attributes: {@code fieldA}, {@code fieldB}, {@code fieldC}, {@code fieldD}. * @return hashCode value */
@Override
public int hashCode() {
int h = 5381;
h += (h << 5) + fieldA.hashCode();
h += (h << 5) + fieldB.hashCode();
h += (h << 5) + fieldC.hashCode();
h += (h << 5) + fieldD.hashCode();
return h;
}
/** * Prints the immutable value {@code NewModel} with attribute values. * @return A string representation of the value */
@Override
public String toString() {
return "NewModel{"
+ "fieldA=" + fieldA
+ ", fieldB=" + fieldB
+ ", fieldC=" + fieldC
+ ", fieldD=" + fieldD
+ "}";
}
/** * Creates an immutable copy of a {@link NewModel} value. * Uses accessors to get values to initialize the new immutable instance. * If an instance is already immutable, it is returned as is. * @param instance The instance to copy * @return A copied immutable NewModel instance */
public static ImmutableNewModel copyOf(NewModel instance) {
if (instance instanceof ImmutableNewModel) {
return (ImmutableNewModel) instance;
}
return ImmutableNewModel.builder()
.from(instance)
.build();
}
/** * Creates a builder for {@link ImmutableNewModel ImmutableNewModel}. * <pre> * ImmutableNewModel.builder() * .fieldA(String) // required {@link NewModel#fieldA() fieldA} * .fieldB(Date) // required {@link NewModel#fieldB() fieldB} * .fieldC(Long) // required {@link NewModel#fieldC() fieldC} * .addFieldD|addAllFieldD(String) // {@link NewModel#fieldD() fieldD} elements * .build(); * </pre> * @return A new ImmutableNewModel builder */
public static ImmutableNewModel.Builder builder() {
return new ImmutableNewModel.Builder();
}
/** * Builds instances of type {@link ImmutableNewModel ImmutableNewModel}. * Initialize attributes and then invoke the {@link #build()} method to create an * immutable instance. * <p><em>{@code Builder} is not thread-safe and generally should not be stored in a field or collection, * but instead used immediately to create instances.</em> */
@Generated(from = "NewModel", generator = "Immutables")
public static final class Builder {
private static final long INIT_BIT_FIELD_A = 0x1L;
private static final long INIT_BIT_FIELD_B = 0x2L;
private static final long INIT_BIT_FIELD_C = 0x4L;
private long initBits = 0x7L;
private String fieldA;
private Date fieldB;
private Long fieldC;
private List<String> fieldD = new ArrayList<String>();
private Builder() {
}
/** * Fill a builder with attribute values from the provided {@code NewModel} instance. * Regular attribute values will be replaced with those from the given instance. * Absent optional values will not replace present values. * Collection elements and entries will be added, not replaced. * @param instance The instance from which to copy values * @return {@code this} builder for use in a chained invocation */
public final Builder from(NewModel instance) {
Objects.requireNonNull(instance, "instance");
fieldA(instance.fieldA());
fieldB(instance.fieldB());
fieldC(instance.fieldC());
addAllFieldD(instance.fieldD());
return this;
}
/** * Initializes the value for the {@link NewModel#fieldA() fieldA} attribute. * @param fieldA The value for fieldA * @return {@code this} builder for use in a chained invocation */
public final Builder fieldA(String fieldA) {
this.fieldA = Objects.requireNonNull(fieldA, "fieldA");
initBits &= ~INIT_BIT_FIELD_A;
return this;
}
/** * Initializes the value for the {@link NewModel#fieldB() fieldB} attribute. * @param fieldB The value for fieldB * @return {@code this} builder for use in a chained invocation */
public final Builder fieldB(Date fieldB) {
this.fieldB = Objects.requireNonNull(fieldB, "fieldB");
initBits &= ~INIT_BIT_FIELD_B;
return this;
}
/** * Initializes the value for the {@link NewModel#fieldC() fieldC} attribute. * @param fieldC The value for fieldC * @return {@code this} builder for use in a chained invocation */
public final Builder fieldC(Long fieldC) {
this.fieldC = Objects.requireNonNull(fieldC, "fieldC");
initBits &= ~INIT_BIT_FIELD_C;
return this;
}
/** * Adds one element to {@link NewModel#fieldD() fieldD} list. * @param element A fieldD element * @return {@code this} builder for use in a chained invocation */
public final Builder addFieldD(String element) {
this.fieldD.add(Objects.requireNonNull(element, "fieldD element"));
return this;
}
/** * Adds elements to {@link NewModel#fieldD() fieldD} list. * @param elements An array of fieldD elements * @return {@code this} builder for use in a chained invocation */
public final Builder addFieldD(String... elements) {
for (String element : elements) {
this.fieldD.add(Objects.requireNonNull(element, "fieldD element"));
}
return this;
}
/** * Sets or replaces all elements for {@link NewModel#fieldD() fieldD} list. * @param elements An iterable of fieldD elements * @return {@code this} builder for use in a chained invocation */
public final Builder fieldD(Iterable<String> elements) {
this.fieldD.clear();
return addAllFieldD(elements);
}
/** * Adds elements to {@link NewModel#fieldD() fieldD} list. * @param elements An iterable of fieldD elements * @return {@code this} builder for use in a chained invocation */
public final Builder addAllFieldD(Iterable<String> elements) {
for (String element : elements) {
this.fieldD.add(Objects.requireNonNull(element, "fieldD element"));
}
return this;
}
/** * Builds a new {@link ImmutableNewModel ImmutableNewModel}. * @return An immutable instance of NewModel * @throws java.lang.IllegalStateException if any required attributes are missing */
public ImmutableNewModel build() {
if (initBits != 0) {
throw new IllegalStateException(formatRequiredAttributesMessage());
}
return new ImmutableNewModel(fieldA, fieldB, fieldC, createUnmodifiableList(true, fieldD));
}
private String formatRequiredAttributesMessage() {
List<String> attributes = new ArrayList<>();
if ((initBits & INIT_BIT_FIELD_A) != 0) attributes.add("fieldA");
if ((initBits & INIT_BIT_FIELD_B) != 0) attributes.add("fieldB");
if ((initBits & INIT_BIT_FIELD_C) != 0) attributes.add("fieldC");
return "Cannot build NewModel, some of required attributes are not set " + attributes;
}
}
private static <T> List<T> createSafeList(Iterable<? extends T> iterable, boolean checkNulls, boolean skipNulls) {
ArrayList<T> list;
if (iterable instanceof Collection<?>) {
int size = ((Collection<?>) iterable).size();
if (size == 0) return Collections.emptyList();
list = new ArrayList<>();
} else {
list = new ArrayList<>();
}
for (T element : iterable) {
if (skipNulls && element == null) continue;
if (checkNulls) Objects.requireNonNull(element, "element");
list.add(element);
}
return list;
}
private static <T> List<T> createUnmodifiableList(boolean clone, List<T> list) {
switch(list.size()) {
case 0: return Collections.emptyList();
case 1: return Collections.singletonList(list.get(0));
default:
if (clone) {
return Collections.unmodifiableList(new ArrayList<>(list));
} else {
if (list instanceof ArrayList<?>) {
((ArrayList<?>) list).trimToSize();
}
return Collections.unmodifiableList(list);
}
}
}
}
Enter fullscreen mode Exit fullscreen mode
Conclusion
The Java language can be even more verbose if we don’t use the proper tools for the job, and for years, code generation has been a useful solution to make our life easier in the Java ecosystem. Reaching a good level of immutability in our codebases requires much effort when doing it manually, and it’s susceptible to inadvertent mistakes, to avoid that, and make our codebase also smaller (less code fewer bugs) we can should the Immutables.org library.
Immutability in Java (2 Part Series)
1 Immutability in Java Made Easy
2 ️ Immutables/AutoValue/Lombok Which One?
暂无评论内容