DEV Community

Marcelo Mendes
Marcelo Mendes

Posted on • Edited on

Reflections em Java: Uma Ferramenta Poderosa

Vamos explorar o que é Reflections, seus usos e riscos, e como aplicá-lo na prática.

O que é Reflection?

Reflection é uma funcionalidade poderosa do Java que permite que o código inspecione e manipule classes, campos, métodos e construtores de objetos, mesmo que sejam privados ou desconhecidos em tempo de compilação.

Essa habilidade é muito boa para cenários dinâmicos, como frameworks e ferramentas que interagem com classes desconhecidas ou que dependem de comportamento fora do default. Porém, Reflection deve ser usada com cuidado devido a seus impactos em desempenho e segurança.


Quando usar Reflection?

Frameworks e libs

Muitas libs utilizam Reflection para automatizar tarefas. Por exemplo, o JUnit usa Reflection para localizar e executar métodos de teste automaticamente.

Interoperabilidade e extensibilidade

Quando você precisa trabalhar com tipos desconhecidos em tempo de compilação. Um exemplo comum é carregar dinamicamente classes em um sistema modular.

Manipulação de objetos externos ou desconhecidos

É útil para acessar métodos ou campos de classes de terceiros que não possuem uma API pública.

Criação de ferramentas e depuração

Reflection é frequentemente usada em ferramentas como depuradores, analisadores de código e geradores de código dinâmico.

Cuidados ao usar Reflection

Reflection é poderosa, mas traz riscos porque pode quebrar algumas garantias fornecidas pelo compilador e pelo runtime no uso default. Aqui estão os principais riscos:

Quebra de encapsulamento

Reflection permite acessar campos e métodos private ou protected, ignorando os modificadores de acesso.

Erros em tempo de execução

Ao contrário do uso tradicional, erros de tipos ou métodos inexistentes só aparecem em tempo de execução, isso dificulta a depuração e os testes.

Impacto no desempenho

Métodos refletivos, como Method.invoke, têm um overhead maior porque bypassam as ações do compilador. Esse impacto pode ser significativo em aplicações com chamadas frequentes ou alta performance.


Ferramentas principais da classe Reflection

O pacote java.lang.reflect oferece várias classes e métodos que tornam possível manipular dinamicamente os elementos de uma classe. Aqui estão algumas das ferramentas mais importantes:

Class: O ponto de partida

A classe Class é usada para obter informações e manipular classes. É frequentemente a porta de entrada para outras operações de Reflection.

Métodos úteis:

getDeclaredFields(): Retorna todos os campos declarados na classe.
getDeclaredMethods(): Retorna todos os métodos declarados na classe.
getDeclaredConstructors(): Retorna todos os construtores declarados na classe.
getSuperclass(): Obtém a superclasse da classe.
getInterfaces(): Retorna as interfaces implementadas pela classe.


Na prática

Suponha que você tenha a seguinte entidade Weather (faltou os getters rs):

Entidade weather que tem varias cidades, um construtor vazio, um construtor com todos args os getters dessas cidades

Essa entidade é atualizada por um cronjob que faz chamadas a uma API externa e salva as temperaturas. Seu objetivo é buscar dinamicamente a temperatura de uma cidade. Veja como fazer isso usando Reflection:

o metodo currentTemperature que receb uma string city e utiliza reflections para acessar os getters dinamicamente e retornar o valor de acordo.

Fluxo

Construção do Nome do Método:

O nome do método é criado dinamicamente concatenando "get" com o nome da cidade e "Temperature".

Obtenção do Método:

Usamos getDeclaredMethod para obter o método correspondente, que nesse caso são os getters.

Invocação do Método:

O método é invocado dinamicamente com invoke, retornando a temperatura.

Benefícios

O uso de Reflection neste cenário evita a necessidade de código repetitivo e cria uma solução flexível para obter dados de diferentes cidades, tornando o código mais modular.


Conclusão

Nesse artigo eu espero que vocês consigam entender de modo geral como essa ferramenta é poderosa no ecossistema Java, permitindo resolver problemas dinâmicos de maneira diferente. Porém, devido à quebra de encapsulamento, impacto no desempenho e dificuldade de depuração, deve ser usada com cuidado.
Entender os cenários adequados para aplicá-la e conhecer seus riscos é essencial para tirar o melhor proveito dessa funcionalidade sem comprometer a manutenção e a segurança do código.

Top comments (1)