Padrão de Projeto Builder em Java: Uma Abordagem Detalhada

O padrão de projeto Builder em Java oferece uma maneira elegante e flexível de construir objetos imutáveis e complexos passo a passo.

Ele separa a criação de um objeto em etapas distintas, permitindo maior controle sobre o processo de construção e promovendo um código mais legível e manutenível.

Parece complicado de entender? Fique tranquilo, é bem simples. Vamos entender como esse design pattern tão importante funciona.

1. O que são Padrões de Projeto? São Realmente Importantes?

Os padrões de projeto, ou design patterns, são formas padronizadas de escrever código.

Se você já programa ou está dando os primeiros passos na programação, já deve ter visto como os arquivos com linhas de código podem ficar extensos.

Dependendo do tamanho do projeto e dos recursos, o código-fonte pode crescer muito, tornando difícil manter a organização.

Foi a partir de problemas como esse que surgiram os padrões de projeto. Eles foram desenvolvidos para organizar o código, tornando-o mais fácil de entender, manter e reutilizar.

Esses padrões se alinham perfeitamente com os pilares da programação orientada a objetos: encapsulamento, herança, abstração e polimorfismo.

Esses aspectos demonstram a relevância e a importância de usar padrões de projeto em seus projetos.

O padrão de projeto Builder pode ser usado em diversas linguagens orientadas a objetos, não se limitando apenas a Java, que será a linguagem abordada neste conteúdo.

O Builder pode ser utilizado em linguagens como C# e PHP, por exemplo.

2. Entendendo e Aplicando o Padrão de Projeto Builder

Agora que entendemos os benefícios de utilizar padrões, vamos compreender como se aplica o padrão Builder.

Para isso, vamos a um cenário simples para uma melhor compreensão.

Imagine o seguinte: você tem uma classe Cliente que possui os atributos nome, cpfCnpj, endereço, telefone e email.

Utilizaremos poucos atributos para simplificar, mas normalmente uma classe Cliente teria muitos atributos.

O padrão de projeto Builder facilita o gerenciamento de classes com muitos atributos.

2.1 Classe Cliente: Atributos e Construtor

public class Cliente {
    private String nome;
    private String cpfCnpj;
    private String endereco;
    private String telefone;
    private String email;

    // Construtor privado para obrigar o uso do Builder
    private Cliente(Builder builder) {
        this.nome = builder.nome;
        this.cpfCnpj = builder.cpfCnpj;
        this.endereco = builder.endereco;
        this.telefone = builder.telefone;
        this.email = builder.email;
    }

Primeiramente, criamos a classe Cliente com os atributos e o construtor.

É importante que o construtor seja privado para que a classe Cliente não possa ser instanciada diretamente de fora.

Dessa forma, centralizamos a instanciação da classe através da classe Builder, que criaremos em seguida.

2.1 Classe ClienteBuilder: Atributos e Métodos

// Classe estática Builder
    public static class ClienteBuilder {
        private String nome;
        private String cpfCnpj;
        private String endereco;
        private String telefone;
        private String email;

        // Métodos para definir os atributos e retornar o Builder
        public Builder nome(String nome) {
            this.nome = nome;
            return this;
        }

        public Builder cpfCnpj(String cpfCnpj) {
            this.cpfCnpj = cpfCnpj;
            return this;
        }

        public Builder endereco(String endereco) {
            this.endereco = endereco;
            return this;
        }

        public Builder telefone(String telefone) {
            this.telefone = telefone;
            return this;
        }

        public Builder email(String email) {
            this.email = email;
            return this;
        }

        // Método para construir a instância de Cliente
        public Cliente build() {
            return new Cliente(this);
        }
    }

A classe buider é criada dentro da classe Cliente como se fosse um método da classe.

Normalmente a classe Builder é estática para facilitar a chamada sem a necessidade de instancia-la.

Na classe ClienteBuilder é utilizado os mesmos atributos da classe pai Cliente, em segunda é criado os métodos que recebe o valor de cada a atributo e retorna todo o objeto com a palavra reservada this que referencia o objeto do escopo da classe ClienteBuilder.

O método build é muito importante para que esse padrão aconteça. vamos ver ele mais uma vezes e detalhas sua importância

// Método para construir a instância de Cliente
        public Cliente build() {
            return new Cliente(this);
        }

Esse método retorna a classe Cliente, construída de acordo com o retorno do objeto completo em cada método do tipo ClienteBuilder. Em outras palavras, os valores de cada atributo foram armazenados no objeto (this) e em seus respectivos atributos para serem retornados na instância de um novo Cliente.

Devemos criar os getters e setters que recebem os valores dos atributos na instância da classe ClienteBuilder, que faz parte da classe Cliente.

No entanto, como o Builder já atribui e retorna os valores dos atributos, podemos criar apenas os getters, pois o Builder já realiza a função dos setters.

// Getters para os atributos
    public String getNome() {
        return nome;
    }

    public String getCpfCnpj() {
        return cpfCnpj;
    }

    public String getEndereco() {
        return endereco;
    }

    public String getTelefone() {
        return telefone;
    }

    public String getEmail() {
        return email;
    }

 @Override
    public String toString() {
        return "Cliente{" +
                "nome='" + nome + '\'' +
                ", cpfCnpj='" + cpfCnpj + '\'' +
                ", endereco='" + endereco + '\'' +
                ", telefone='" + telefone + '\'' +
                ", email='" + email + '\'' +
                '}';

Com a classe Buider criada fica bem simples atribuir e retornar os atributos da classe

Exemplo:

public static void main(String[] args) {
        Cliente cliente = new Cliente.Builder()
                .nome("João da Silva")
                .cpfCnpj("123.456.789-00")
                .endereco("Rua das Flores, 123")
                .telefone("555-1234")
                .email("joao.silva@example.com")
                .build();

        System.out.println(cliente);
    }

3. Vantagens do Padrão de Projeto Builder

O uso do padrão de projeto Builder em Java oferece diversas vantagens. É importante conhecer essas vantagens em detalhes para saber em quais projetos é viável aplicá-lo.

Alguns projetos são tão simples que a aplicação de um padrão como o Builder pode não ser necessária, ou pode ser que outro padrão resolva melhor os requisitos do projeto.

3.1 Maior Legibilidade e Manutenabilidade

O código fica mais organizado e fácil de entender, pois a lógica de criação de objetos está separada em métodos distintos.

Isso facilita a leitura e a manutenção do código, pois cada método tem uma responsabilidade clara e bem definida.

3.2 Flexibilidade

O padrão de projeto Builder permite construir objetos com diferentes configurações, definindo apenas os atributos necessários.

Isso é particularmente útil quando se trabalha com objetos complexos que têm muitos atributos opcionais ou interdependentes.

3.3 Imutabilidade

Os objetos construídos com o padrão de projeto Builder são imutáveis por padrão, promovendo segurança e confiabilidade.

Uma vez que o objeto é criado, seus atributos não podem ser alterados, o que evita bugs relacionados a modificações inesperadas no estado do objeto.

3.4 Encadeamento de Métodos

A construção de objetos pode ser feita de maneira fluida através do encadeamento de métodos de configuração. Isso torna o código mais conciso e legível, permitindo que você configure todos os atributos necessários em uma única expressão.

3.5 Reutilização

A classe Builder pode ser reutilizada para criar diferentes tipos de objetos, o que promove a reutilização de código e reduz a duplicação.

Isso é especialmente útil em aplicações complexas onde várias classes compartilham uma estrutura de construção similar.

Conclusão

O padrão de projeto Builder é uma ferramenta poderosa e flexível para a criação de objetos complexos em Java.

Ele oferece uma maneira elegante de separar a lógica de construção de objetos, promovendo código mais legível, manutenível e seguro.

Ao utilizar o padrão Builder, você pode construir objetos de maneira flexível e fluida, definindo apenas os atributos necessários e garantindo a imutabilidade do objeto final.

Implementar o padrão Builder em seus projetos Java pode trazer inúmeros benefícios, especialmente em cenários onde a criação de objetos é uma tarefa complexa e suscetível a erros.

Com um bom entendimento desse padrão, você estará melhor equipado para escrever código mais robusto e sustentável.

Por isso, se você ainda não está utilizando o padrão Builder em seus projetos, considere adotá-lo e aproveite todas as vantagens que ele tem a oferecer.

Marcos R.S
Marcos R.S

Olá, pessoal! Sou Marcos, apaixonado por aprender, especialmente sobre tecnologia. Estou sempre em busca de lapidar os conhecimentos que já possuo e adquirir novos. Atuo com análise e desenvolvimento de sistemas, sou graduando em Sistemas de Informação e tenho formação técnica em Informática.

Deixe um comentário

O seu endereço de e-mail não será publicado. Campos obrigatórios são marcados com *