Skip to content

Latest commit

 

History

History
341 lines (235 loc) · 8.67 KB

05-lists.adoc

File metadata and controls

341 lines (235 loc) · 8.67 KB

Secuencias, Listas y Rangos

La secuencia es la clase padre de List, String, Map, Range y cualquier otra que implemente el protocolo de iteración. La característica principal de una secuencia es que esta podría ser virtualmente infinita (sin elementos que indiquen un fin). Entre los métodos que podemos destacar de Sequence están:

each

Similar a un bucle for. Por cada elemento dentro de la secuencia llama a una función cuyo parámetro es el siguiente elemento en la secuencia.

var numeros = [1, 2, 3]
numeros.each {|num|
  System.print(num)
}

map

Llama a una función cuyo parámetro es el siguiente elemento en la secuencia. Retorna una nueva secuencia con el resultado de las operación en cada elemento. El resultado de map es almacenado por referencia. Lo que significa que cambios posteriores en la secuencia original pueden afectar el valor del resultado de map. Para pasarlo a una variable independiente, se debe ejecutar el método toList.

var numeros = [1, 2, 3]

// .toList es necesario para que System.print() muestre la lista apropiadamente.
// Esto es debido a que una secuencia puede no tener fin. Necesitamos pasarlo a una estructura de datos finita.
// También estamos utilizando el "retorno implícito".
numeros = numeros.map{|num| num + 1}.toList

// Muestra [2, 3, 4]
System.print(numeros)

where

Filtra una secuencia. Entregará una nueva lista la cual cumpla la condicional. Al igual que map se debe ejecutar el método toList para evitar que modificaciones en la secuencia original afecten la nueva secuencia.

var numeros = [1, 2, 3, 4]

var pares = numeros.where{|num| num % 2 == 0}.toList

// Muestra [2, 4]
System.print(pares)

reduce

Llama a una función cuyo parámetro es el siguiente elemento en al secuencia. Retorna un único elemento, producto de las operaciones en cada elemento.

var numeros = [1, 2, 3]

// Si no se pasa un acumulador, se considera el primer elemento como valor inicial
var resultado = numeros.reduce {|acumulador, num| acumulador + num}

// Muestra 6
System.print(resultado)

// Iniciamos el acumulador con valor -1
resultado = numeros.reduce(-1) {|acumulador, num| acumulador + num}

// Muestra 5
System.print(resultado)

take

Obtiene un número de elementos dentro de una secuencia.

var numeros = [1, 2, 3, 4]
var primeros = numeros.take(2).toList

// Muestra [1, 2]
System.print(primeros)

skip

La operación inversa de take. Salta un número determinado de elementos.

var numeros = [1, 2, 3, 4]
var ultimos = numeros.skip(2).toList

// Muestra [3, 4]
System.print(ultimos)

contains

Retorna verdadero si la secuencia contiene al elemento.

var numeros = [1, 2, 3]

// Muestra "false" debido a que no contiene el número 4
System.print(numeros.contains(4))

count

Retorna el número de elementos de la secuencia. También es posible solamente contar elementos que cumplan una condición.

var numeros = [1, 2, 3, 4]

// 4
System.print(numeros.count)

// Contamos solamente los números múltiplos de 2
var pares = numeros.count {|num| num % 2 == 0}

// 2
System.print(pares)

join

Une a los elementos de una secuencia transformándolos en un String. Si no se pasa un parámetro se asume un caracter vació como unión.

var numeros = [1, 2, 3]

// Muestra "123"
System.print(numeros.join())

// Muestra "1$2$3"
System.print(numeros.join("$"))

// Muestra "1 num 2 num 3"
System.print(numeros.join(" num "))

all

Itera sobre los elementos de una secuencia con una condicional. Devuelve verdadero si todos los elementos pasan la condición. Si alguno de los elementos es falso, termina de iterar y devuelve falso.

var numeros = [1, 2, 3]
var resultado = numeros.all {|num| num > 2}

// Falso. No todos los números en la secuencia son mayores a 2
System.print(resultado)

resultado = numeros.all{|num| num < 4}

// Verdadero. Todos los números en la secuencia son menores a 4
System.print(resultado)

any

Similar a all. Devuelve verdadero al primer elemento que pase la condición. Si ninguno de los elementos pasa la condición, devuelve falso.

var numeros = [1, 2, 3]
var resultado = numeros.any {|num| num % 2 == 0}

// Verdadero. Existe un número múltiplo de 2.
System.print(resultado)

resultado = numeros.any {|num| num % 5 == 0}

// Falso. Ningún número es múltiplo de 5.
System.print(resultado)

ejemplo

El siguiente ejemplo utiliza algunas funciones de Sequence para obtener el primer elemento de una secuencia dependiendo de una condición.

  // Creamos una nueva función para nuestra lógica
  var primerElemento = Fn.new {|elementos, predicado|

    // Si existe un elemento que cumpla la condición, obtenemos el primero de la secuencia.
    var resultado = elementos.any(predicado) ? elementos.where(predicado).take(1) : false
    if (resultado && resultado.count > 0) {
      return resultado.toList[0]
    }

    // Caso contrario devolvemos nulo
    return null
  }

  var numeros = [1, 2, 3]
  var resultado = primerElemento.call(numeros) {|item| item > 1}

  // Muestra 2
  System.print(resultado)

  resultado = primerElemento.call(numeros) {|item| item > 3}

  // Muestra null
  System.print(resultado)

Los Subíndices nos ayudan a obtener valores dentro de una lista, mapa u objeto que implemente el método de subíndice utilizando los paréntesis cuadrados (corchetes) [].

lista[0]    // Obtiene el primer elmento de la lista
mapa["id"] // Obtiene el valor asociado con "id"

El siguiente código muestra como implementar un objeto que acepta dos parámetros en su método de subíndice.

class Matriz {
  [x, y] {"x %(x), y %(y)"}
  construct nueva() {}
}

var matriz = Matriz.nueva()

// Muestra x 1 y 1
System.print(matriz[1,1])

Una lista es una colección de elementos que son almacenados en una variable. Los elementos pueden estar relacionados de cierta forma o no tener ningún tipo de relación.

El siguiente ejemplo muestra algunas operaciones disponibles para listas:

var estudiantes = ["Pedro", "Javier", "Nicolás"]

// Iteramos sobre los elementos de la lista
for (estudiante in estudiantes) {
    System.print("¡Hola, " + estudiante + "!")
}

// La misma operación anterior, pero utilizando la función each
estudiantes.each{|estudiante|
  System.print("¡Hola, " + estudiante + "!")
}

// Añadimos un nuevo estudiante en la última posición
estudiantes.add("Juan")

// Muestra ["Pedro", "Javier", "Nicolás", "Juan"]
System.print(estudiantes)

// Muestra "false" por que no existe el estudiante María
System.print(estudiantes.contains("María"))

// Muestra ["Pedro", "Javier"]
System.print(estudiantes.take(2).toList)

// Insertamos un nuevo elemento en la última posición disponible
estudiantes.insert(estudiantes.count - 1, "Javier Manzana")

// Filtramos todos los estudiantes que se llamen Javier
// Muestra: ["Javier", "Javier Manzana"]
System.print(estudiantes.where {|nombre| nombre.startsWith("Javier")}.toList)

// Elimina todos los estudiantes
estudiantes.clear()
// Generamos una lista de 4 elementos con valor 1
var elementos = [1] * 4

// Muestra [1, 1, 1, 1]
System.print(elementos)

También podemos combinar listas fácilmente

var numeros = [1, 2, 3]
var letras = ["a", "b", "c"]
var elementos = numeros + letras

// Muestra [1, 2, 3, a, b, c]
System.print(elementos)

Un rango es un objeto que representa una lista finita, incremental e iterable de números. Es una operación disponible en cada instancia de un número. Es dada por la operación .. y la operación …​.

  • inicio..fin: Incluye al número fin. Ejemplo: 1..3 retorna [1, 2, 3].

  • inicio…​fin: No incluye al número fin. Ejemplo 1…​3 retorna [1, 2].

// Mostramos los números del 1 al 5
(1..5).each {|num|
  System.print("Número %(num)")
}
// Mostramos del 3 al 1
((1 + 2)..(3 - 2)).each {|num|
  System.print("Número %(num)")
}
// Mostramos del 1 al 3 utilizando bucle for.
for (num in 1..3) {
  System.print("Número %(num)")
}

Los rangos son muy útiles para obtener subconjuntos dentro de objetos iterables como Strings y Lists.

var nombre = "Juanito"

// Muestra "nito"
System.print(nombre[3..-1])

// Muestra "otinauJ"
System.print(nombre[-1..0])
var letras = ["a", "b", "c", "d", "e"]
var subconjunto = letras[1..3]

// Muestra [b, c, d]
System.print(subconjunto)