Method Chaining: In java, Method Chaining is used to invoke multiple methods on the same object which occurs as a single statement. Method-chaining is implemented by a series of methods that return the this reference for a class instance.
Implementation: As return values of methods in a chain is this reference, this implementation allows us to invoke methods in chain by having the next method invocation on the return value of the previous method in the chain.
Java
final class Student {
private int id;
private String name;
private String address;
public Student setId( int id)
{
this .id = id;
return this ;
}
public Student setName(String name)
{
this .name = name;
return this ;
}
public Student setAddress(String address)
{
this .address = address;
return this ;
}
@Override public String toString()
{
return "
id = "
+ this .id + "
, name = "
+ this .name + "
, address = "
+ this .address;
}
}
public class MethodChainingDemo {
public static void main(String args[])
{
Student student1 = new Student();
Student student2 = new Student();
student1.setId( 1 )
.setName(" Ram & quot;)
.setAddress(" Noida & quot;);
student2.setId( 2 )
.setName(" Shyam & quot;)
.setAddress(" Delhi & quot;);
System.out.println(student1);
System.out.println(student2);
}
}
|
Output:
id = 1, name = Ram, address = Noida
id = 2, name = Shyam, address = Delhi
Need of Builder Pattern: Method chaining is a useful design pattern but however if accessed concurrently, a thread may observe some fields to contain inconsistent values. Although all setter methods in above example are atomic, but calls in the method chaining can lead to inconsistent object state when the object is modified concurrently. The below example can lead us to a Student instance in an inconsistent state, for example, a student with name Ram and address Delhi.
Output may be:
id = 2, name = Shyam, address = Noida
Another inconsistent output may be
id = 0, name = null, address = null
Note : Try running main method statements in loop(i.e. multiple requests to server simultaneously). To solve this problem, there is Builder pattern to ensure the thread-safety and atomicity of object creation. Implementation : In Builder pattern, we have a inner static class named Builder inside our Server class with instance fields for that class and also have a factory method to return an new instance of Builder class on every invocation. The setter methods will now return Builder class reference. We will also have a build method to return instances of Server side class, i.e. outer class.
Java
final class Student {
private final int id;
private final String name;
private final String address;
public Student(Builder builder)
{
this .id = builder.id;
this .name = builder.name;
this .address = builder.address;
}
public static class Builder {
private int id;
private String name;
private String address;
public static Builder newInstance()
{
return new Builder();
}
private Builder() {}
public Builder setId( int id)
{
this .id = id;
return this ;
}
public Builder setName(String name)
{
this .name = name;
return this ;
}
public Builder setAddress(String address)
{
this .address = address;
return this ;
}
public Student build()
{
return new Student( this );
}
}
@Override
public String toString()
{
return "id = " + this .id + ", name = " + this .name +
", address = " + this .address;
}
}
class StudentReceiver {
private volatile Student student;
public StudentReceiver()
{
Thread t1 = new Thread( new Runnable() {
@Override
public void run()
{
student = Student.Builder.newInstance()
.setId( 1 )
.setName("Ram")
.setAddress("Noida")
.build();
}
});
Thread t2 = new Thread( new Runnable() {
@Override
public void run()
{
student = Student.Builder.newInstance()
.setId( 2 )
.setName("Shyam")
.setAddress("Delhi")
.build();
}
});
t1.start();
t2.start();
}
public Student getStudent()
{
return student;
}
}
public class BuilderDemo {
public static void main(String args[])
{
StudentReceiver sr = new StudentReceiver();
System.out.println(sr.getStudent());
}
}
|
Output is guaranteed to be one of below:
id = 1, name = Ram, address = Noida
OR
id = 2, name = Shyam, address = Delhi
OR
id = 2, name = Ram, address = Noida
The Builder.newInstance() factory method can also be called with any required arguments to obtain a Builder instance by overloading it. The object of Student class is constructed with the invocation of the build() method. The above implementation of Builder pattern makes the Student class immutable and consequently thread-safe. Also note that the student field in client side code cannot be declared final because it is assigned a new immutable object. But it be declared volatile to ensure visibility of shared reference to immutable objects. Also private members of Builder class maintain encapsulation. Please have a look at append method of StringBuilder class in java.lang package to understand implementations of Builder pattern more.
Further Read: Java Design Patterns Tutorial
Whether you're preparing for your first job interview or aiming to upskill in this ever-evolving tech landscape,
GeeksforGeeks Courses are your key to success. We provide top-quality content at affordable prices, all geared towards accelerating your growth in a time-bound manner. Join the millions we've already empowered, and we're here to do the same for you. Don't miss out -
check it out now!