Transforme arrays em ouro: aprenda a manipular dados com o map() em Javascript
O map é um método Javascript que cria uma array a partir de outra, sem alterar a original.
Fim.
Isso é tudo que você precisa saber.
Daqui em diante o que você vai ler é apenas essa explicação inicial em detalhes.
Veja como gerar uma array de strings em caixa alta, a partir de outra em caixa baixa:
const lowercaseNames = [ 'goku', 'bulma', 'kuririn' ];
const uppercaseNames = lowercaseNames.map(function(name) {
return name.toUpperCase();
});
console.log(uppercaseNames); // [ 'GOKU', 'BULMA', 'KURIRIN' ]
E agora, como gerar uma array com o dobro dos números da array original:
const numbers = [ 5, 8, 3, 6, 7, 2, 5 ];
const double = numbers.map(function(number) {
return number * 2;
});
console.log(double); // [ 10, 16, 6, 12, 14, 4, 10 ]
Sempre explico alguns termos da programação aqui no blog com a tradução dos comandos.
Mas que diabos é "mapear"?
Fiz um post completo no meu blog com toda a explicação de como usar o map em 90% dos casos.
Vou deixar o link aqui nas fontes :D
Se puder, me dê um feedback de o que achou do contéudo!
Muito bom! Complementando, acho que a principal lição é que map
sempre retorna outro array, então se vc não precisa desse outro array, não deveria usar map
.
Tenho visto muitas pessoas usando map
indevidamente como substituto de for
/forEach
, o que apesar de "funcionar", é um uso torto. Por exemplo:
const array = [1,2,3,4];
// ATENÇÃO: uso indevido de map, pois um for seria mais adequado
array.map(n => console.log(n * 2));
O código até "funciona" (imprime os valores desejados), mas é um uso torto por vários motivos:
map
retorna outro array, que no caso não foi usado pra nada. Ou seja, gerou outro array à toa.- semanticamente fica confuso, pois o uso de
map
sugere que os valores serão mapeados e deseja-se ter/usar o outro array com os resultados. Mas como não foi isso que aconteceu, acaba causando confusão, principalmente se outras pessoas forem dar manutenção no código ("Por que não usou umfor
?")
Neste caso seria muito mais adequado um loop:
for (const n of array) {
console.log(n * 2);
}
Ou o forEach
:
array.forEach(n => console.log(n * 2));
Lembrando ainda que forEach
(assim como map
, filter
, reduce
, etc) recebe uma função de callback que é executada para cada elemento. Claro que para poucos arrays pequenos não faz diferença, mas tem que lembrar que chamadas de função têm seu custo, que pode ou não ser desprezível dependendo do caso.
Quanto a um dos seus exemplos, vale lembrar que ao retornar o mesmo objeto, uma alteração neste causa uma mudança em todos os arrays:
const characters = [
{ name: 'Goku', race: 'saiyajin' },
{ name: 'Cell', race: 'android' },
{ name: 'Kuririn', race: 'terráqueo' },
{ name: 'Bulma', race: 'terráqueo' },
{ name: 'Vegeta', race: 'saiyajin' },
{ name: 'Mestre Kame', race: 'terráqueo' },
{ name: 'Kami-Sama', race: 'namekuseijin' }
];
const mapeado = characters.map(character => {
if (character.race === 'terráqueo')
return {
name: character.name.toUpperCase(),
race: character.race
};
return character;
});
// nome "Goku" não foi alterado
console.log(mapeado[0]); // { name: 'Goku', race: 'saiyajin' }
// muda o nome no array original
characters[0].name = 'GOKU';
// a alteração se reflete no outro array, já que ele tem uma referência para o mesmo objeto
console.log(mapeado[0]); // { name: 'GOKU', race: 'saiyajin' }
Como o objeto referente a "Goku" é o mesmo nos dois arrays, qualquer alteração feita em um reflete no outro.
Para evitar isso, teria que retornar uma cópia mesmo, então bastaria mudar para:
const mapeado = characters.map(character => {
if (character.race === 'terráqueo')
return {
name: character.name.toUpperCase(),
race: character.race
};
return { ...character };
});
Lembrando que neste caso estou fazendo uma shallow copy, ou seja, se o objeto character
tiver outros objetos dentro ele, o problema persistiria. Neste caso, teria que ser feita uma deep copy (dê uma olhada aqui para mais detalhes).
Por fim, a função também poderia ser assim:
const mapeado = characters.map(character => {
const novo = { ...character };
if (character.race === 'terráqueo')
novo.name = character.name.toUpperCase();
return novo;
});
Esta abordagem seria mais vantajosa caso cada objeto tivesse muitas propriedades, pois aí vc já copia todas de uma vez. Neste caso não faz tanta diferença porque são só duas (name
e race
), mas se tivesse mais, seria bem massante ficar copiando uma a uma.