Como ya hemos visto en el tema anterior, son funciones que tienen como argumento una función o incluso devuelven una función.
Funciones como variable
Por otra parte, también podemos guardar funciones como variables en Kotlin aprovechándonos de los datos de tipo Función que implementa su gramática.
var suma = { x: Int, y: Int -> x+ y }
El bloque de código anterior es completamente válido y además nos permite cambiar el bloque de código que se ejecuta, por ejemplo, en cada iteración.
DEFINICIÓN
En base a lo visto anteriormente, podemos definir el concepto de función lambda como:
Una función lambda es un literal de función que puede ser usado como expresión. Esto quiere decir, una función que no está ligada a un identificador y que puedes usar como valor.
O, en otras palabras, una función anónima que utilizamos como valor de entrada en una función de orden superior.
SINTÁXIS
Continuando con la función suma, podemos escribirla así:
funsuma(x: Int, y: Int) = x + y
Para transformarla al formato lambda tenemos que hacerla anónima.
La sintaxis de un literal de lambda va escrita entre llaves {...} y su contenido es el siguiente:
Lista de parámetros — Cada parámetro es una declaración de variable, aunque esta lista es opcional
Operador de flecha -> — Se omite si no usas lista de parámetros
Cuerpo del lambda — Son las sentencias que van luego del operador de flecha
{ x: Int, y: Int -> x+ y}
Función lambda con cuerpo
Las funciones lambda tienen que devolver un valor, eso es indispensable. Sin embargo, el cuerpo de una función lambda puede tener varias sentencias:
// Función lambda para hayar la potencia de x elevado a y.{ x: Int, y: Int -> var res =1for (i in1..y) res *= x res}
IDENTIFICADOR IT
Cuando tu lambda usa un único argumento y no piensas cambiar su nombre por cuestiones de legibilidad, puedes usar el identificador it.
Esta variable se deduce implícitamente con el tipo inferido por el compilador y puedes referirte a ella como tu parámetro.
Vamos a mostrar un caso de uso con todo lo que hemos visto anteriormente:
En el caso del tipo Array, el argumento que coge la lambda es el iterador. Por tanto, it será el número de iteración.
ACCESO A VARIABLES EXTERNAS
Las lambdas pueden acceder a variables que se hayan inicializado fuera del código de la lambda:
funrecorrerArray(array: Array<Int>, fn: (Int) -> Unit) {for (i in array) {fn(i) }}funmain() {val array3 =Array(5) { it *2 }var suma =0recorrerArray(array3) { suma += it }println(suma) // 20}
EJEMPLOS DE APLICACIÓN
Las lambdas tienen un caso de uso claro con las funciones de orden superior como lo que vamos a ver a continuación:
// Declaración de función de orden superiorfuncalculadora(x: Int, y: Int, fn: (Int, Int) -> Int): Int {returnfn(x, y)}// uso con lambdasprintln("La suma de $x y $y es: ${calculadora(x, y, { x: Int, y: Int -> x+ y })}")println("La resta de $x y $y es: ${calculadora(x, y, { x: Int, y: Int -> x- y })}")println("La multiplicación de $x por $y es: ${calculadora(x, y, { x: Int, y: Int -> x* y })}")println("La división de $x entre $y es: ${calculadora(x, y, { x: Int, y: Int -> x/ y })}")println("La potencia de $x elevado a $y es: ${calculadora(x, y,{ x: Int, y: Int -> var res =1for (i in1..y) res *= x res})}")/* ResultadoLa suma de 5 y 5 es: 10La resta de 5 y 5 es: 0La multiplicación de 5 por 5 es: 25La división de 5 entre 5 es: 1La potencia de 5 elevado a 5 es: 3125 */
Omitir el paréntesis
Sin embargo, las lambdas por definición se pueden poner fuera de los paréntesis lo que hace el código más legible:
println("La suma de $x y $y es: ${calculadora(x, y) { x: Int, y: Int -> x+ y }}")
Ejemplo por partes
Ejemplo básico
// Implementación de una variable que almecena la cantidad de 'i' en una String.val s ="Supercalifragilisticoespialidoso"val iCounter = s.count({ char:Char -> char=='i' })println("En la frase: $s") // En la frase: Supercalifragilisticoespialidosoprintln("Hay $iCounter ies.") // Hay 6 ies.
Ejemplo omitiendo los paréntesis
// La lambda se puede sacar de los paréntesisval s ="Supercalifragilisticoespialidoso"val iCounter = s.count() { char:Char -> char=='i' }// Los paréntesis vacíos se pueden omitirval s ="Supercalifragilisticoespialidoso"val iCounter = s.count { char:Char -> char=='i' }
Ejemplo omitiendo el tipo de dato
// El tipo de dato se puede omitir ya que count se aplica solo a tipo char.val s ="Supercalifragilisticoespialidoso"val iCounter = s.count { char -> char =='i' }
Ejemplo del identificador it
En el ejemplo anterior lo que hemos hecho ha sido renombrar el argumento como char. Eso se puede omitir tambien.
// Se puede omitir el argumento de manera explícita.// De esta manera el argumento se llamará it implícitamente:val s ="Supercalifragilisticoespialidoso"val iCounter = s.count { it =='i' }