Skip to main content

what is immutable and how to make a class immutable in java

 Immutability is a fundamental concept in Java that plays a crucial role in ensuring the predictability and safety of your code. An immutable class is a class whose instances cannot be modified after they are created.

In other words, the state of an immutable object remains constant throughout its lifetime. This concept has significant benefits in terms of thread safety, debugging, and code quality. In this blog post, we'll explore what immutability is and how to create an immutable class in Java, step by step.

Step 1: Declare the Class as final

The first step in creating an immutable class is to declare it as final. This prevents any other class from extending it, which is a crucial aspect of ensuring immutability.

java
public final class ImmutableClass { // Class members and methods go here }

Step 2: Make Fields Private and Final

All fields in an immutable class should be declared as private and final. Making them private restricts access from external sources, and marking them final ensures that their values cannot be changed after initialization.

java
public final class ImmutableClass { private final int field1; private final String field2; // Other fields go here public ImmutableClass(int field1, String field2) { this.field1 = field1; this.field2 = field2; // Initialize other fields here } }

Step 3: Provide Accessor Methods

To access the values of the fields, provide getter methods for each field. These methods allow clients to read the values but not modify them.

java
public final class ImmutableClass { private final int field1; private final String field2; public ImmutableClass(int field1, String field2) { this.field1 = field1; this.field2 = field2; } public int getField1() { return field1; } public String getField2() { return field2; } }

Step 4: Ensure Mutable Objects Are Handled Correctly

If your immutable class contains references to mutable objects (e.g., collections), make sure they are handled properly. You should return a copy of the mutable object rather than the object itself in getter methods to avoid unintentional changes.

java
public final class ImmutableClass { private final List<String> immutableList; public ImmutableClass(List<String> immutableList) { this.immutableList = new ArrayList<>(immutableList); // Create a defensive copy } public List<String> getImmutableList() { return new ArrayList<>(immutableList); // Return a copy to keep the original list immutable } }

Step 5: Prevent Subclassing

To ensure the class remains immutable, prevent subclassing by declaring all constructors as private and providing a factory method to create instances.

java
public final class ImmutableClass { private final int field1; private final String field2; private ImmutableClass(int field1, String field2) { this.field1 = field1; this.field2 = field2; } public static ImmutableClass createInstance(int field1, String field2) { return new ImmutableClass(field1, field2); } public int getField1() { return field1; } public String getField2() { return field2; } }

Step 6: Make Defensive Copies When Necessary

If your class deals with mutable objects, make defensive copies to ensure that the original objects are not modified outside the class.

java
public final class ImmutableClass { private final List<String> immutableList; private ImmutableClass(List<String> immutableList) { this.immutableList = new ArrayList<>(immutableList); } public static ImmutableClass createInstance(List<String> immutableList) { return new ImmutableClass(immutableList); } public List<String> getImmutableList() { return new ArrayList<>(immutableList); } }

By following these steps, you can create a robust immutable class in Java. This ensures that the instances of your class remain constant, providing safety and predictability in your code.

Example Code:

Here's a complete example of an immutable class in Java:

java
public final class ImmutableClass { private final List<String> immutableList; private ImmutableClass(List<String> immutableList) { this.immutableList = new ArrayList<>(immutableList); } public static ImmutableClass createInstance(List<String> immutableList) { return new ImmutableClass(immutableList); } public List<String> getImmutableList() { return new ArrayList<>(immutableList); } }

By following these guidelines, you can create a class that guarantees immutability and the associated benefits it provides.

Comments

Popular posts from this blog

Using Java 8 Streams to Find the Second-Highest Salary in an Employee List

To find the second-highest salary from a list of employees using Java 8 streams, you can follow these steps: Create a list of employees with their salaries. Use Java 8 streams to sort the employees by salary in descending order. Skip the first element (which is the employee with the highest salary). Get the first element of the remaining stream (which is the employee with the second-highest salary). Example code: java import java.util.ArrayList; import java.util.List; class Employee { private String name; private double salary; public Employee (String name, double salary) { this .name = name; this .salary = salary; } public double getSalary () { return salary; } } public class SecondHighestSalary { public static void main (String[] args) { List<Employee> employees = new ArrayList <>(); employees.add( new Employee ( "John" , 60000.0 )); employees.add( new Employe...

Top 20 Exception Handling Interview Questions and Answers for Experienced Java Developers

Introduction: Exception handling is a crucial aspect of Java development, ensuring robust and error-tolerant code. Experienced Java developers are expected to have a deep understanding of exception handling mechanisms. In this blog post, we'll explore the top 20 interview questions related to exception handling, accompanied by detailed answers and sample code snippets to help you prepare for your next Java interview. 1. What is an exception in Java? An exception is an event that disrupts the normal flow of a program. In Java, exceptions are objects that represent errors or abnormal situations during runtime. java try { // Code that may throw an exception } catch (ExceptionType e) { // Code to handle the exception } 2. Differentiate between checked and unchecked exceptions. Checked exceptions are checked at compile-time, and the programmer is forced to either catch them or declare that the method throws them. Unchecked exceptions, on the other hand, are not checked at ...

Mastering Java Streams: A Complete Guide with Examples and Interview Questions

Java Streams have revolutionized the way data processing tasks are handled in Java programming. Introduced in Java 8, Streams offer a fluent and functional approach to processing collections of objects. In this guide, we'll delve into what Streams are, how they work, and provide practical examples along the way. Understanding Java Streams: Java Streams represent a sequence of elements that can be processed sequentially or in parallel. They provide a pipeline through which data can be manipulated using various operations such as filtering, mapping, sorting, and aggregating. Benefits of Java Streams: Concise and Readable Code : Streams promote a functional programming style, leading to more concise and readable code compared to traditional imperative approaches. Lazy Evaluation : Stream operations are lazily evaluated, meaning elements are processed only when necessary, improving efficiency. Parallelism : Streams can leverage parallel processing for improved performance on multicore ...

Subscribe to get new posts

Name

Email *

Message *