Princípio da inversão de dependências - Dependency Inversion Principle

Ao ser aplicado permite que os detalhes passem a depender de abstrações, respeitando a direção da regra de dependências.

Categoria de Programação

Postado em 27 abril 2022

Atualizado em 07 junho 2023

Palavras-chave: dip,solid,principle,princípios,dependecy,inversion,programming

Visualizações: 2142

A regra de dependências diz que os detalhes devem depender das abstrações. Na arquitetura limpa, essa é a regra básica. Módulos de nível alto não devem saber sobre módulos de nível baixo. Se uma classe sabe sobre outra, ela passa a depender dela. Depender de uma classe significa sofrer influência dessa classe. Na arquitetura limpa, as regras de negócio (abstrações) não devem sofrer influência de detalhes (implementações).

Porém, pode ser difícil obedecer a regra de dependências. Em certas ocasiões, as regras de negócio precisam saber sobre os detalhes. O princípio de inversão de dependências (Dependency Inversion Principle) pode ser aplicado para mudar a direção das dependências e fazer com que a regra de dependência seja obedecida.

O que é o princípio de inversão de dependências (DIP)?

O princípio de inversão de dependências é o quinto princípios dos princípios SOLID. Esse princípio diz que a regra de dependências deve ser respeitada. Ou seja:

Os detalhes devem depender de abstrações

Esse princípio também diz que módulos de nível baixo devem depender de módulos de nível alto e nunca ao contrário. As abstrações se referem a interfaces e classes abstratas. Pelo fato das abstrações possuirem apenas uma estrutura de métodos não implementados, o comportamento desses métodos podem ser modificados na implementação. Classes que implementam a mesma abstração podem ser substituídas facilmente.

interface Abstracao {
    void abstractMethod();
}

class ClasseA implements Abstracao {
	public void abstractMethod() {
		// Implementa um tipo de comportamento
	}
}

class ClasseB implements Abstracao {
	public void abstractMethod() {
		// Implementa outro tipo de comportamento
	}
}

As classes que usam o método abstractMethod() não sabem sobre os detalhes da implementação, apenas sabem que elas podem usar o método pré-definido.

class ClasseC {
    Abstracao abstracao;
    
    public ClasseC(Abstracao abstracao) {
	    this.abstracao = abstracao;
    }
	
	public void usarAbstracao() {
		this.abstracao.abstractMethod();
	}
}

A classeC acima, definiu a abstração como parâmetro. Ela não se importa se uma instância da classe B ou C estão sendo passadas desde que as classes implementem a interface “abstracao”. Nesse caso, podemos afirmar que a classeC depende de abstrações.

exemplo de princípio de inversão de dependências

O exemplo acima já está usando o DIP. Por isso, o módulo de nível baixo aponta para o módulo de nível alto e os detalhes dependem da abstração. Perceba que a classeC pode usar o método que os detalhes implementam de modo indireto. Além disso, a classeC não sabe nada sobre os detalhes, respeitando a regra de dependências.

Como usar o princípio de inversão de dependências (DIP)?

O princípio de inversão de dependências só tem um objetivo: inverter as dependências. Aplicamos o DIP quando as dependências não estão respeitando a regra de dependências.

exemplo de violação do princípio de inversão de dependências

O diagrama acima mostra um exemplo de violação de DIP. O módulo de nível alto não deve saber sobre o módulo de nível baixo. Em outras palavras, a classeAA não deve usar a classeZZ.

A regra de dependências diz que módulo de nível baixo devem apontar (depender) para módulos de nível baixo. A violação acima pode ser resolvida com a adição de uma abstração.

exemplo de como resolver uma violação do princípio de inversão de dependências

Aplicando o DIP, as dependências agora apontam para as abstrações.

Por que usar o princípio de inversão de dependências (DIP)?

A principal vantagem ao aplicar o princípio de inversão de dependências é a reutilização. No exemplo acima, quando o DIP era violado, dificilmente a classe ZZ poderia ser substituída por outra sem afetar o módulo de nível alto. Na realidade, existiria uma cadeia de dependências transitivas que usam a classe ZZ, o que tornaria essa substituição mais difícil ainda. Outra grande desvantagem dessa violação é que ela viola outro princípio, o princípio aberto fechado.

Ao resolver a violação acima usando o DIP, as classes que implementam a abstração podem ser substituídas por outras classes que implementam a mesma abstração sem afetar o módulo de nível alto. A classe que pertence ao módulo de nível alto não sabe nada sobre os detalhes que pertencem ao módulo de nível baixo. Além de proteger o módulo de nível alto, esse princípio ajuda na manutenção e possibilita a aplicação do princípio aberto fechado de modo automático.

Conclusão

O princípio de inversão de dependências inverte as dependências para que elas possam obedecer a regra das dependências. Na arquitetura limpa, os detalhes devem apontar para as abstrações para possibilitar a aplicação do princípio aberto fechado, possibilitar a reutilização e melhorar a manutenção.

Projetos práticos

Desenvolvendo o campo de visão de um personagem em um plano 2D

Detectando objetos que entram dentro do campo de visão do personagem. Útil para servir de "gatilho" para eventos em um jogo.

Criando um jogo de guerra nas estrelas em javascript usando a biblioteca p5.js

Jogo simples de guerra espacial desenvolvido em javascript. Esse jogo usa cálculos de física para simular efeitos de atrito e inércia.

Criando um jogo de pacman usando javascript e pixi.js (parte 1)

Desenvolvimento dos conceitos mais básicos do clássico pacman, como: mapa, animação, deslocamento e detector de colisões.

Criando o esqueleto de um jogo de tiro 2D visto de cima usando P5.js

Usando lógicas matemáticas como trigonometria para criar e calcular o esqueleto de um jogo de tiro 2D em javascript

Integrando o PHP com Elasticsearch no desenvolvimento de um sistema de busca

Projeto de criação de um sistema de busca usando o framework Symfony e Elasticsearch. A integração com Kibana também é feito de modo remoto com um raspberrypi.

Veja também

O endereçamento de dispositivos na internet é automatizado graças ao DHCP

Antigamente o endereçamento de dispositivos era feito manualmente, porém isso traz muitas dificuldades em questão de administração. O DHCP resolve esses problemas

Os três pilares de segurança da informação são o requerimento mínimo para um serviço seguro

A quantidade de programadores só tende a aumentar com o tempo. Porém, muitos programadores ainda não conhecem os três pilares de segurança da informação.

Algoritmo

O algoritmo é um conjunto de instruções escritas por um programador com intuito de solucionar um problema ou obter um resultado previsto.

Princípio de substituição de Liskov - Liskov Substitution Principle

Esse princípio diz que uma classe derivada deve ser substituível pela sua classe base sem apresentar comportamentos inesperados.

Princípio de segregação de interfaces - Interface Segregation Principle

Os clientes não devem ser forçados a importar métodos que eles não usam. Os métodos devem ser segregados de modo abstrato em interfaces.

Programação orientada a objetos

A programação orientada a objetos possui um contexto bastante semelhante com a vida real, facilitando a sua implementação e interpretação.