====== Interfaces funcionales, lambdas y referencias a métodos ======
==== Interfaces funcionales ====
  * Las interfaces funcionales son interfaces que poseen un único método abstracto (sin implementar)
  * Aparecen por primera vez en Java 8
  * Están especialmente preparadas para usarse con expresiones lambda
  * Pueden tener definidos métodos por defecto (tienen implementación)
En Java existen una serie de interfaces funcionales ya definidas especialmente útiles
  * **Predicate**: Evalua una condición booleana
    * //IntPredicate//, //LongPredicate//, //DoublePredicate//
  * **Consumer**: Acepta un input como entrada y no devuelve ningún resultado
    * //IntConsumer//, //LongConsumer//, //DoubleConsumer//
  * **Function**: Acepta un input como entrada y devuelve un resultado
    * //IntFunction//, //IntToDoubleFunction//, //IntToLongFunction//, //LongFunction//, LongToDoubleFunction, . . .
  * **Supplier**: No acepta argumentos y devuelve un resultado (al contrario que //Consumer//)
    * //BooleanSupplier//, //IntSupplier//, //LongSupplier//, //DoubleSupplier//
  * **BiPredicate**: Versión de //Predicat//e para dos argumentos
  * **BiFunction**: Versión de //Function// para dos argumentos
Predicate checkEmpty = (value) -> value.isEmpty();
System.out.println(checkEmpty.test("hola")); 
System.out.println(checkEmpty.test(""));
Pero también podemos crear las nuestras propias:
@FunctionalInterface
public interface SayMyName {
  void say(String name); 
}
==== Expresiones lambda ====
  * Aparecen por primera vez en Java 8
  * Código más compacto y limpio
  * Sustituyen al concepto de clase anónima
  * Permiten expresar las instancias de interfaces funcionales de una forma compacta
  * Su uso está especialmente extendido en el uso de Streams, donde se pueden utilizar de forma que el código queda mucho más compacto y fácil de entender
  * Cuando usamos una tenemos que imaginarnos que estamos implementando el código necesario para ejecutar luego el método que tiene por defecto asignada esa interface
La sintaxis es:
(parametros) -> (cuerpo-lambda)
Por ejemplo:
// Para ordenar una lista
listaAnimales.sort((animal1, animal2) -> (animal1.getPeso() – animal2.getPeso());
// Para crear un Predicate que permita evaluar una colección sobre un objeto
Animal unAnimal = new Animal(. . . .)
Predicate pesoMayorQue10 = (animal) -> animal.getPeso() > 10; 
pesoMayorQue10.test(unAnimal); // true/false
==== Referencias a métodos ====
Suponiendo que tenemos una colección de cadenas, veamos unos ejemplos que nos permitan utilizar una referencia a un método:
List cadenas = new ArrayList<>();
=== Métodos estáticos ===
// Utilizando una expresión lambda
cadenas.forEach((cadena) -> System.out.println(cadena))
// Equivalente utilizando referencia a método
cadenas.forEach(System.out::println)
=== Métodos de instancia ===
// Suponemos que existe una clase CadenasComparator que implementa Comparator
// Utilizando una expresión lambda
cadenas.sort((cadena1, cadena2) -> cadenasComparator.compare(cadena1, cadena2))
// Equivalente utilizando una referencia a método suponiendo que existe un objeto de la clase que implementa a CadenasComparator
cadenas.sort(cadenasComparator::compare);
// Equivalente utilizando una referencia a método
cadenas.sort(CadenasComparator::compare);
=== Constructores ===
CadenasComparator::new
----
(c) 2019-{{date>%Y}} Santiago Faci