Referencia a métodos en expresiones lambda y switch¶
Una referencia a métodos o métodos referenciados proporciona una forma de referirse a un método sin ejecutarlo. Se relaciona con expresiones lambda porque también requiere un contexto de tipo de objetivo que consiste en una interfaz funcional compatible.
En el caso de que todo lo que haga la expresión lambda sea llamar a otro método con los parámetros pasados a la expresión lambda, la implementación de Java lambda proporciona una forma más corta de expresar la llamada al método, que es usando ::
Veamos un ejemplo:
public interface Impresora{
void imprimir(String s);
}
Si usáramos lambda sin referencias a métodos, lo haríamos de la siguiente forma:
Impresora impresora = s -> System.out.println(s);
Dado que todo lo que hace el cuerpo lambda es reenviar el parámetro String al método System.out.println()
, podemos reemplazar la declaración lambda anterior con una referencia de método utilizando
{Clase::Metodo}
De forma que quedaría:
Impresora impresora = System.out::println;
foreach
para que imprima los elementos de la lista
List<String> names = new ArrayList();
names.add("Andrea");
names.add("Luisa");
names.add("Diego");
names.add("Paúl");
names.add("Dario");
names.forEach(System.out::println);
Observa los dos puntos dobles ::
. Estos le indican al compilador de Java que se trata de una referencia de método. El método al que se hace referencia es lo que viene después de los dos puntos dobles. Cualquier clase u objeto que posea el método al que se hace referencia viene antes de los dos puntos dobles.
Podemos hacer referencia a los siguientes tipos de métodos:
- Método estático
- Método con parámetros de objeto
- Método de instancia
- Constructor
Referencias a métodos estáticos¶
Los métodos más fáciles de referenciar son los métodos estáticos. Veamos un ejemplo:
public interface Finder {
int find(String s1, String s2);
}
public class MyClass{
public static int doFind(String s1, String s2){
return s1.lastIndexOf(s2);
}
}
La referencia al método estático doFind
se haría con:
Finder finder = MyClass::doFind;
Dado que los parámetros de los métodos Finder.find()
y MyClass.doFind()
coinciden, es posible crear una expresión lambda que implemente Finder.find()
y haga referencia al método MyClass.doFind()
.
Referencia a métodos con parámetro¶
También puede hacer referencia a un método con parámetros de objeto de la clase al método que se llama.
public interface Finder {
int find(MyClass mc, String s1, String s2);
}
class MyClass {
public int check(String s1, String s2) {
return s1.indexOf(s2);
}
}
//dentro del main
//Aunque es un método de instancia, no estático
//podemos llamarlo con la clase porque en la interfaz
//está añadido un parámetro de tipo MyClass
Finder finder = MyClass::check;
Si quisiéramos hacerlo con una expresión lambda sin usar referencia de métodos que llama a String.indexOf() para buscar sería:
Finder finder = (s1, s2) -> s1.indexOf(s2);
Su equivalente utilizando referencia de métodos con parámetro de objeto en la expresión lambda sería:
Finder finder = String::indexOf;
int numero = finder.find("Hola", "a");
Observe cómo la versión abreviada hace referencia a un solo método. El compilador de Java intentará hacer coincidir el método al que se hace referencia con el primer tipo de parámetro, utilizando el segundo tipo de parámetro como parámetro del método al que se hace referencia.
Referencia a métodos de instancia¶
En tercer lugar, también es posible hacer referencia a un método de instancia desde una definición lambda.
public interface Deserializer {
int deserialize(String v1);
}
Esta interfaz representa un componente que es capaz de "deserializar" un String en un int.
public class StringConverter {
public int convertToInt(String v1){
return Integer.valueOf(v1);
}
}
El método convertToInt()
tiene la misma signatura que el método deserialize()
del método de la interfaz Deserializer
. Por eso, podemos crear una instancia de StringConverter y hacer referencia a su método convertToInt() desde una expresión lambda de Java.
StringConverter stringConverter = new StringConverter();
Deserializador des = stringConverter::convertToInt;
La expresión lambda creada por la segunda de las dos líneas hace referencia al método convertToInt de la instancia de StringConverter creada en la primera línea.
Referencias a constructores¶
Finalmente, es posible hacer referencia a un constructor de una clase. Para ello, se escribe el nombre de la clase seguido de ::new:
-- Nomenclatura
MiClase::new
Supplier<Usuario> usu = Usuario::new;
//Construye un objeto de tipo usuario que es devuelto por método get();
Usuario usuario = usu.get();
Veamos otro ejemplo utilizando BiFunction
:
class Pruebas {
public static void main(String[] args) {
BiFunction<String, Integer, Usuario> crearUsuario = Usuario::new;
Usuario u = crearUsuario.apply("Patricia", 12);
}
}
class Usuario {
private String nombre;
private int edad;
public Usuario(String nombre, int edad) {
this.nombre = nombre;
this.edad = edad;
}
}
Expresiones switch y lambdas¶
Desde Java 12, se ha incluido las expresiones switch que junto las lambdas, nos permite crear código más compacto. En estos momentos switch
, nos permite devolver un valor y asignarlo. Para ello, por cada case
va seguido de una expresión lambda que devuelve un valor. Por otro lado, es obligada la sentencia default
enum Mes {
ENERO, FEBRERO, MARZO, ABRIL, MAYO, JUNIO, JULIO, AGOSTO, SEPTIEMBRE,OCTUBRE, NOVIEMBRE,DICIEMBRE;
}
public static String estacion(Mes mes){
String result=switch (mes){
case DICIEMBRE,ENERO, FEBRERO -> "Invierno";
case MARZO, ABRIL, MAYO -> "Primavera";
case JUNIO, JULIO, AGOSTO -> "Verano";
default -> "Otoño";
};
return result;
}
int day = 3;
String dayName = switch (day) {
case 1 -> "Lunes";
case 2 -> "Martes";
case 3 -> "Miércoles";
case 4 -> "Jueves";
case 5 -> "Viernes";
case 6 -> "Sábado";
case 7 -> "Domingo";
//lanzamos excepción si no es válido el valor
default -> throw new IllegalArgumentException("Día no válido");
};
yield¶
Si la sentencia lambda tiene varias líneas, tenemos que usar yield
para devolver el valor
int nota=5;
Boolean esAprobado=switch (nota){
case 0,1,2,3,4->{
System.out.println("Suspenso");
//devuelve false
yield false;
}
case 5,6,7,8,9,10 -> {
System.out.println("Aprobado");
//devuelve true
yield true;
}
default -> throw new IllegalArgumentException("Nota no válido");
};
instanceof vs switch¶
Por otro lado, ahora, podemos prescindir de instanceof
para determinar el tipo de un objeto. Mediante instanceof
tendríamos
static String formatter(Object o) {
String formatted = "unknown";
if (o instanceof Integer i) {
formatted = String.format("int %d", i);
} else if (o instanceof Long l) {
formatted = String.format("long %d", l);
} else if (o instanceof Double d) {
formatted = String.format("double %f", d);
} else if (o instanceof String s) {
formatted = String.format("String %s", s);
}
return formatted;
}
static String formatterPatternSwitch(Object o) {
return switch (o) {
case null -> System.out.println("null!");
case Integer i -> String.format("int %d", i);
case Long l -> String.format("long %d", l);
case Double d -> String.format("double %f", d);
case String s -> String.format("String %s", s);
default -> o.toString();
};
}