Sunday, December 29, 2024
Google search engine
HomeLanguagesJavaBuilder Pattern in java

Builder Pattern in java

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




// Java code to demonstrate method chaining
 
final class Student {
 
    // instance fields
    private int id;
    private String name;
    private String address;
 
    // Setter Methods
    // Note that all setters method
    // return this reference
    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;
    }
}
 
// Driver class
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




// Java code to demonstrate Builder Pattern
 
// Server Side Code
final class Student {
 
    // final instance fields
    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;
    }
 
    // Static class Builder
    public static class Builder {
 
        /// instance fields
        private int id;
        private String name;
        private String address;
 
        public static Builder newInstance()
        {
            return new Builder();
        }
 
        private Builder() {}
 
        // Setter methods
        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;
        }
 
        // build method to deal with outer class
        // to return outer instance
        public Student build()
        {
            return new Student(this);
        }
    }
 
    @Override
    public String toString()
    {
        return "id = " + this.id + ", name = " + this.name +
                               ", address = " + this.address;
    }
}
 
// Client Side Code
class StudentReceiver {
 
    // volatile student instance to ensure visibility
    // of shared reference to immutable objects
    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;
    }
}
 
// Driver class
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.

RELATED ARTICLES

Most Popular

Recent Comments