Una seqüència és un conjunt d'elements agrupats un a continuació de l'altre.
Introducció
Una manera habitual d'agrupar un conjunt de valor és mitjançant una seqüencia.
array
Typescript té una estructura de dades de tipus []
que es fa servir per guardar els elements que tu vulguis en una "array".
Edita el fitxer main.ts
:
{
const list: number[] = [6, 8, 11]
}
Com que els elements de l'array estan ordenats pots accedir a cada elements per la seva posició dins de l'array.
[ 0 ] => 6
[ 1 ] => 8
[ 2 ] => 11
Només has de tenir en compte que es comença a contar des del número 0
.
Has d'accedir indicant la posició entre claudators:
{
const list: number[] = [6, 8, 11]
console.log(list[0]) // 6
console.log(list[1]) // 8
console.log(list[2]) // 11
}
A continuació tens una llista de fruites:
{
const fruits: string[] = ["Apple", "Orange", "Plum"]
}
Que passa si intentes accedir al cinquè element de la llista?
{
const fruits: string[] = ["Apple", "Orange", "Plum"]
const fruit = fruits[4]
console.log(fruit)
}
Executem el codi:
$ bun main.ts
undefined
Que els valor de la variable fruit
és undefined
perquè la llista només té tres elements.
Si vols pots modificar un element de la llista indicant la seva posició:
{
const fruits: string[] = ["Apple", "Orange", "Plum"]
fruits[1] = "Banana"
console.log(fruits) // [ "Apple", "Banana", "Plum" ]
}
També pots saber la mida d'una llista amb la propietat length
:
{
const fruits: string[] = ["Apple", "Orange", "Plum"]
console.log(fruits.length) // 3
}
Si vols pots afegir elements a una llista:
{
const fruits: string[] = []
fruits.push("Banana")
fruits.push("Apricot")
fruits.push("Cherry")
console.log(fruits) // [ "Banana", "Apricot", "Cherry" ]
}
Els elements sempre s'afegeixen al final de la llista.
Bucle
Una de les operacions més habituals amb un array és recorrer tots els elements de la llista.
Amb for .. of
pots recorrer tots els elements d'una llista:
{
const fruits: string[] = ["Apple", "Orange", "Apricot", "Plum"]
for(const fruit of fruits) {
console.log(fruit)
}
}
A continuació et mostrarem algunes solucions que no són del tot eficients, de moment!
Un codi primer ha de funcionar, després ha de funcionar bé, i per últim funcionar ràpid.
I el més important, un codi s'ha de poder llegir i entendre a la primera.
map
Una operació habitual és transformar una llista aplicant una transformació a cada element de la llista:
A continuació tens un exemple:
{
const xs: number[] = [8, 9, 4, 5, 6]
const ys: number[] = []
for (const x of xs) {
ys.push(x * 2)
}
console.log(ys) // [ 16, 18, 8, 10, 12 ]
}
filter
Una altre operació habitual és filtrar els elements d'una llista.
Per exemple, només vols una llista amb els valors més grans que 5:
{
const xs: number[] = [8, 9, 4, 5, 6]
const ys: number[] = []
for (const x of xs) {
if (x > 5)
ys.push(x)
}
console.log(ys) // [ 8, 9, 6 ]
}
reduce
Un algoritme habitual en llistes es reduir la llista a un sol valor.
Per exemple, pots sumar tots els elements de la llista:
{
const xs: number[] = [8, 9, 4, 5, 6, 2, 9, 4, 7, 2, 3]
let sum = 0 // valor inicial
for (const x of xs) {
sum += x
}
console.log(sum) //50
}
Calcula la mitjana aritmètica de la llista:
{
const xs: number[] = [8, 9, 4, 5, 6, 2, 9, 4, 7, 2, 3]
let sum = 0 // valor inicial
for (const x of xs) {
sum += x
}
const mean = sum / xs.length
console.log(mean) //5.36
}
find
Una altra operació habitual és buscar si una llista té un element.
{
const target = "Apricot"
const fruits: string[] = ["Apple", "Orange", "Apricot", "Plum"]
let result = false
for (const fruit of fruits){
if (fruit == target) {
result = true
break
}
}
console.log(result) // true
}
Activitat
1.- Tens una llista xs
de tipus number[]
.
Busca el valor màxim de la llista
{
const xs: number[] = [8, 9, 4, 5, 6, 2, 9, 4, 7, 2, 3]
// ...
console.log(max) // 9
}
{
const xs: number[] = [8, 9, 4, 5, 6, 2, 9, 4, 7, 2, 3]
let max = xs[0] // valor inicial
for (const x of xs) {
if (x > max)
max = x
}
console.log(max) // 9
}
2.- A continuació tens una llista de persones:
{
const xs: string[] = ["Olivia Anderson", "Ethan Carter", "Isabella Brooks",
"Olivia Bennett", "Ava Morgan", "Noah Mitchell", "Mia Harper", "Liam Parker"]
}
Crea una nova llista només amb les persones que el seu nom és Olivia
.
"Olivia Bennet".startsWith("Olivia") // true
{
const xs: string[] = ["Olivia Anderson", "Ethan Carter", "Isabella Brooks",
"Olivia Bennett", "Ava Morgan", "Noah Mitchell", "Mia Harper", "Liam Parker"]
const result: string[] = []
for (const x of xs) {
if (x.startsWith("Olivia"))
result.push(x)
}
console.log(result) // [ "Olivia Anderson", "Olivia Bennett" ]
}
3.- Molts cops tens una llista de notes en que et falta alguna nota perque l'alumne no ha fet l'examen.
Calcula la mitjana aritmètica d'aquesta llista:
{
const xs: (number| null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
// ...
console.log(mean) //5.36
}
{
const xs: (number| null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
// Creem una llista nova sense els valors null
const ys: number[] = []
for (const x of xs) {
if (x == null) // si un element és null no el processem
continue
ys.push(x)
}
// Sumem tots els valors de la llista filtrada
let sum = 0 // valor inicial
for (const y of ys) {
sum += y
}
// Calculem la mitjana aritmètica
const mean = sum / ys.length
console.log(mean) //5.36
}
Llistes imbricades
Una llista pot estar formada per llistes:
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
}
Per processar aquesta llista necessites escriure dos bucles.
Per exemple, pots afegir tots els números en una sola llista:
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
const ys: number[] = []
for (const xs of xss) {
for (const x of xs) {
ys.push(x)
}
}
console.log(ys) // [3, 4, 5, 2, 3, 8, 9, 1, 4, 9, 1, 1, 2, 0, 9]
}
Activitat
1.- Crea una llista amb les sumes de cada llista.
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
// ...
console.log(ys) // [ 25, 24, 12 ]
}
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
const ys: number[] = []
for (const xs of xss) {
let y = 0
for (const x of xs) {
y += x
}
ys.push(y)
}
console.log(ys) // [ 25, 24, 12 ]
}
2.- Crea una llista amb la mitja aritmètica de cada llista.
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
// ...
console.log(ys) // [ 4.16, 4.8, 3 ]
}
{
const xss: number[][] = [
[3, 4, 5, 2, 3, 8],
[9, 1, 4, 9, 1],
[1, 2, 0, 9]
]
const ys: number[] = []
for (const xs of xss) {
let sum = 0
for (const x of xs) {
sum += x
}
ys.push(sum / xs.length)
}
console.log(ys) // [ 4.16, 4.8, 3 ]
}
}
Slice
La funció slice(start,end)
torna una vista parcial de la llista.
{
const xs: string[] = ["ant", "bison", "camel", "duck", "elephant"]
const ys = xs.slice(0, 2)
console.log(ys) // [ "ant", "bison" ]
}
El segon paràmetre és opcional:
{
const xs: string[] = ["ant", "bison", "camel", "duck", "elephant"]
const ys = xs.slice(3)
console.log(ys) // [ "duck", "elephant" ]
}
També pots utilitzar index negatius (comencen a contar al revés de dreta a esquerra):
{
const xs: string[] = ["ant", "bison", "camel", "duck", "elephant"]
const ys = xs.slice(1, -1)
console.log(ys) // [ "bison", "camel", "duck" ]
}
La funció slice
és útil quan només vols processar un conjunt d'elements de la llista.
{
const xs: number[] = [3, 4, 5, 2, 3, 8, 9, 1, 4, 9, 1, 1, 2, 0, 9]
let sum = 0
for (const x of xs.slice(0,3)) {
sum += x
}
console.log(sum) // 12
}
Funcions
A Funcions veurem que són les funcions "arrow".
Però és important que entenguis perquè treballem amb les llistes com hem estat fent fins ara.
La majoria de les vegades les llistes no es processen amb for .. of
, sinó que es processen amb funcions de manera funcional.
Les operacions bàsiques són: map
, filter
, reduce
i find
.
Amb la funció map
pots aplicar una funció a cada element de la llista creant una nova llista:
{
const xs: number[] = [8, 9, 4, 5, 6]
const ys = xs.map(x => x * 2)
console.log(ys) // [ 16, 18, 8, 10, 12 ]
}
Amb la funció reduce
pots sumar tots els elements d'una llista:
{
const xs: number[] = [8, 9, 4, 5, 6]
const sum = xs.reduce((acc, x) => acc + x, 0) // acc: accumulator
console.log(sum) // 32
}
Amb la funció filter
pots filtrar els elements d'una llista:
{
const xs: (number | null)[] = [8, 9, null, 4, 5, null, 6]
const ys = xs.filter(value => value != null)
console.log(ys) // [ 8, 9, 4, 5, 6 ]
}
Pots aplicar les funcions una darrera l'altre:
{
const xs: (number | null)[] = [8, 9, null, 4, 5, null, 6]
const ys = xs.
filter(value => value != null).
map(x => x * 5).
reduce((acc, x) => acc + x)
console.log(ys) // 160
}
Activitats
1.- Calcula la mitjana aritmètica dels 5 primers valors de la llista:
{
const xs: (number| null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
// ...
console.log(mean) // 5.2
}
{
const xs: (number | null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
let sum = 0
let n = 0
for (const x of xs.slice(0, 5)) {
if (x == null)
continue
sum += x
n += 1
}
console.log(sum / n) // 6.5
}
2.- Calcula la mitjana aritmètica dels 5 primers números de la llista (has d'ometre els valors null
):
{
const xs: (number| null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
// ...
console.log(mean) // 5.2
}
{
const xs: (number| null) [] = [8, 9, 4, 5, null, 6, 2, 9, null, 4, 7, 2, 3]
const mean = xs.filter(x => x!= null).slice(0,5).reduce((acc,x)=> acc + x) / 5
console.log(mean) // 6.4
}
3.- A continuació tens una llista de lectures de temperatures de l'observatori "Can Pixa" que necessita urgentment un nou lector de temperatures.
Calcula la mitja de la temperatura de cada dia filtrant aquelles temperatures que són clarament errònes:
{
const tss = [
[23, null, 45, 666, 45, null],
[3, -4, 5, null, 4555555555, 6],
[5, 4, 23, null, -333333, 6]
]
}
{
const tss = [
[23, null, 45, 666, 45, null],
[3, -4, 5, null, 4555555555, 6],
[5, 4, 23, null, -333333, 6]
]
const result = []
for (const ts of tss) {
let sum = 0
let n = 0
for (const t of ts) {
if (t == null || t < -30 || t > 90 )
continue
sum += t
n+=1
}
result.push(sum / n) // [ 37.666666666666664, 2.5, 9.5 ]
}
console.log(result)
}
4.-. Escriu un codi que vagi preguntat a l'usuari un número fins que encerti algun de la llista.
Has de donar pistes a l'usuari!
{
const xs: number[] = Array.from({ length: 100 },
() => Math.floor(Math.random() * 100000))
console.log(xs)
}
const xs: number[] = Array.from({length: 100},
() => Math.floor(Math.random() * 100000))
let found = false
while (!found) {
const guess = Number(prompt("?"))
let pista = Number.MAX_VALUE
for (const x of xs) {
if (guess == x) {
console.log("Ben Fet!")
found = true
break
}
let tmp = x - guess
if (tmp < pista) {
pista = tmp
}
}
if (!found) {
pista -= Math.floor(Math.random() * pista)
console.log(pista)
}
}
5.- A continuació tens una llista de llistes de números.
Escriu un codi que torni la llista en què la suma dels seus números és menor:
{
const xss: number[][] = Array.from({ length: 100 }, () =>
Array.from({ length: 100 },
() => Math.floor(Math.random() * 100000)))
console.log(xss)
}
const xss: number[][] = Array.from({length: 100}, () =>
Array.from({length: 100},
() => Math.floor(Math.random() * 100000)))
let result
let min = Number.MAX_VALUE
for (const xs of xss) {
let sum = xs.reduce((a, b) => a + b, 0)
if (sum < min) {
result = xs
min = sum
}
}
console.log(`Llista: ${result}`)
console.log(`Suma: ${min}`)