O Java 8 trás um novo pacote chamado de java.util.function que contém uma série de interfaces que podem e devem ser reaproveitadas para a utilização de expressões lambdas, essas interfaces possuem apenas um método abstrato e esta característica faz com que elas sejam definidas como interfaces funcionais. Algumas interfaces antigas como Runnable e ActionListerner apesar de não sofrerem nenhum tipo de alteração na versão 8 da linguagem, também passaram a ser definidas como tal.
Criando interfaces funcionais
Para criar uma interface funcional é muito simples, basta criar um interface comum que contenha um único método abstrato, a partir daí sua interface já poderá ser usada com uma expressão lambda, vamos a um exemplo:
Temos ainda a opção de anotar a nossa interface para que ela seja explicitamente uma interface funcional, desta forma:
Isto garante que caso a interface ganhe um novo método abstrato acidentalmente, esta excessão seja lançada:
Exception in thread "main" java.lang.Error: Unresolved compilation problem:
The target type of this expression must be a functional interface
E fará com que a sua IDE informe que o programa contém erros. Esta anotação é opcional por um motivo muito importante: garantir que as interfaces existentes antes do Java 8 possam herdar as características das mais novas.
4 minutos para ler
|
java e tutoriais
|
19 Jul, 2015
Seja bem-vindo a esta série de tutoriais sobre o Java 8. Aqui abordaremos as principais funcionalidades acrescentadas nesta nova versão da linguagem e suas vantagens. Estaremos construindo diversos exemplos práticos que nos auxiliarão no aprendizado destes novos conceitos que o Java implementou.
Uma das principais características implementada na linguagem foi a expressão lambda, com ela podemos instanciar classes anônimas que possuem apenas um único método abstrato. Vamos analisar uma nova forma de fazer iteração em coleções:
No exemplo acima temos uma lista de strings e em seguida exibimos seu conteúdo de duas formas distintas: a primeira com um for each tradicional e a segunda utilizando um método chamado forEach, que encontra-se na interface java.lang.Iterable na qual List é herdada, seu argumento é do tipo java.util.function.Consumer uma interface que contem um único método abstrato, o accept(T t).
Nos próximos artigos iremos entender como foi possível a interface Iterable ganhar esse novo método sem que tenha quebrado a compatibilidade com versões anteriores da linguagem.
No método forEach passamos como parâmetro a expressão lambda s -> System.out.println(s), que é uma maneira mais concisa de escrever:
O s antes do símbolo -> é equivalente ao parâmetro do método accept, já o que vem depois do símbolo é equivalente ao corpo do método. Temos ainda outras formas de escrever as expressões lambdas, como apresentadas nos exemplos abaixo:
Podemos notar que:
Quando o método não possui argumentos como é o caso do run da interace Runnable, temos que usar () antes do símbolo de flecha.
Para um corpo que possui mais de uma instrução, devemos coloca-lo entre chaves.
Para casos que o método possui dois ou mais argumentos, temos a sintaxe (a, b) ->, os parenteses só podem ser omitidos quando temos um único argumento.
Outro detalhe sobre as expressões lambdas é que podemos utilizar varáveis locais do bloco em que a expressão está contida, assim como ocorre com as classe anônimas (lembre que existe uma exigência para que a variável seja final).
A partir do Java 8 você não precisa mais explicitar que a variável local é final para utilizar dentro de classes anônimas ou expressões lambdas, basta apenas não alterá-la que o compilador irá traduzir como final.
Conclusão
As expressões lambdas são alternativas mais concisas para instanciar classes anônimas, tornando o código mais enxuto e menos verboso. Quando aliadas aos novos recursos da linguagem teremos mecanimos para tornar o nosso código mais simples e fáceis de serem lidos. No próximo artigo entederemos o que são interaces funcionais e como elas são importantes para as novas APIs da linguagem.