Sistemas complexos requerem padrões de código bem formulados para garantir extensibilidade, manutenibilidade, flexibilidade, fácil entendimento e qualidade. Cada linguagem de programação é construída em cima de um paradigma diferente que, muitas vezes, tenta resolver esse problema de formas distintas. Porém, cabe a pessoa desenvolvedora ter conhecimento desses conceitos para o bom uso dessas ferramentas.
Nessa seção focaremos em alguns padrões e práticas de código mais usados. É importante notar que alguns desses conceitos derivam do paradigma de orientação a objeto e outros do paradigma funcional, e funcionam melhor em contextos específicos. Porém, aqui, veremos cada conceito separadamente, pois cada software tem requisitos diferentes.
Imutabilidade é um conceito muito presente em linguagens funcionais, que buscam garantir que uma entidade sempre contenha o mesmo valor, possibilitando até que a funcionalidade ali definida possa ser provada matematicamente. Para que isso seja verdade, é preciso garantir que funções não mantenham estado e não tenham valores variáveis (ou seja, sejam imutáveis).
Porém, esse conceito tem evoluído além de linguagens funcionais, pois imutabilidade evita também que a sua aplicação retorne informações conflitantes, causadas por concorrência, que fazem a aplicação difícil de debugar.
Quando Alan Kay criou o SmallTalk por volta de 1970, ele acreditava que cada objeto era uma representação de todos os recursos do computador, sendo, dessa forma, completamente independente. É nesse contexto que se formou a ideia de encapsulamento. Os seus componentes devem ser simples o suficiente para poderem ser transportados por toda a aplicação, sem perderem suas propriedades.
- [Wikipedia] Encapsulamento
- [Artigo] Erlang Is the Most Object Oriented Language 🇬🇧
- [Artigo] Curiosidade - Early History of SmallTalk (história do SmallTalk) 🇬🇧
Polimorfismo é a propriedade de ter uma só interface para vários tipos. Ou seja, uma funcionalidade tomaria várias formas.
Pode ser tanto implementado como uma interface ou classe abstrata em Java, ou por herança em muitas linguagens orientadas a objeto.
Linguagens dinamicas, como Ruby, oferecem ainda mais flexibilidade para polimorfismo, dando pouca importancia ao tipo do objeto mas apenas se preocupando que o objeto implemente a funcionalidade desejada. Assim, o código fica mais flexível para, por exemplo, iterar sobre uma lista e chamar um mesmo método sobre objetos diferentes.
SOLID é um acrônimo para os cinco princípios básicos de design de classes orientadas a objetos criados por Robert C. Martin (Uncle Bob). Princípios dão nomes a conceitos complexos sobre o código, mas não são regras ou verdades absolutas.
- Single responsibility principle (princípio de responsabilidade única) O princípio de responsabilidade única define que uma classe ou método deveria existir somente para implementar uma funcionalidade, mas que essa responsabilidade seja completamente encapsulada pela entidade.
- Open/close principle (princípio aberto/fechado) O princípio aberto/fechado dita que entidades de software deveriam ser abertas à extensão, mas fechadas à modificação. A ideia é que essas entidades (Classes, Módulos, Funções, etc.) possam ter suas funcionalidades extendidas por outras classes sem a necessidade de modificar seu código.
- Liskov substitution principle (princípio da substituição de Liskov) Esse princípio visa definir o conceito de subtipo, garantindo que subtipos mantenham as propriedades definidas no tipo original. Esse princípio afirma que se S é um subtipo do objeto T, qualquer objeto de tipo T pode ser substituído por objetos de tipo S.
- Interface segregation principle (princípio de separação de interfaces) ISP diz que nenhum cliente deveria depender de métodos que ele não vai usar. Promove interfaces menores, desacopladas, e expressivas.
- Dependency inversion principle (princípio de inversão de depêndencias) Esse príncipio define que módulos de alto-nível não dependem de módulos de baixo-nível, mas que os dois dependem de abstrações. Além disso, abstrações (como, por exemplo, interfaces) não deveriam depender de detalhes de implementação, mas o contrário.
- [Artigo] The Principles of Object Oriented Design 🇬🇧
- [Artigo] Getting a SOLID start 🇬🇧
- [Wikipedia] SOLID (object-oriented design) 🇬🇧
- [Artigo] SOLID (série de artigos sobre os princípios)
Os padrões de projeto surgiram e surgem devido a solução de problemas recorrentes em aplicações orientadas a objetos. São aplicações diretas dos pilares de orientação a objeto (Encapsulamento, Polimorfismo e Herança). Conhecer padrões de projeto vai ajudar você a entender melhor como esse paradigma funciona e como usar seus poderosos recursos.