title | slug |
---|---|
Arreglos |
Learn/JavaScript/First_steps/Arrays |
{{LearnSidebar}}{{PreviousMenuNext("Learn/JavaScript/First_steps/Useful_string_methods", "Learn/JavaScript/First_steps/Silly_story_generator", "Learn/JavaScript/First_steps")}}
En este último artículo de este módulo, veremos las matrices — una manera ordenada de almacenar una lista de elementos de datos bajo un solo nombre de variable. Aquí vemos por qué esto es útil, luego exploramos cómo crear un arreglo, recuperar, agregar y eliminar elementos almacenados en un arreglo, y más.
Prerrequisitos: | Conocimientos básicos de informática, una comprensión básica de HTML y CSS, una idea de lo que es JavaScript. |
---|---|
Objectivo: | Para entender qué son las matrices y cómo manipularlas en JavaScript. |
Las matrices se describen como "objetos tipo lista"; básicamente son objetos individuales que contienen múltiples valores almacenados en una lista. Los objetos de arreglos pueden almacenarse en variables y tratarse de la misma manera que cualquier otro tipo de valor, la diferencia es que podemos acceder individualmente a cada valor dentro de la lista y hacer cosas útiles y eficientes con la lista, como recorrerlo con un bucle y hacer una misma cosa a cada valor. Tal vez tenemos una serie de productos y sus precios almacenados en un arreglo, y queremos recorrerlos e imprimirlos en una factura, sumando todos los precios e imprimiendo el total en la parte inferior.
Si no tuvieramos matrices, tendríamos que almacenar cada elemento en una variable separada, luego llamar al código que hace la impresión y agregarlo por separado para cada artículo. Esto sería mucho más largo de escribir, menos eficiente y más propenso a errores. si tuviéramos 10 elementos para agregar a la factura, ya sería suficientemente malo, pero ¿ qué pasa con 100 o 1000 artículos? Volveremos a este ejemplo más adelante en el artículo.
Como en artículos anteriores, aprendamos sobre los aspectos básicos reales de las matrices ingresando algunos ejemplos en una consola de JavaScript. A continuación proporcionamos uno (también puedes abrir en consola en una pestaña o ventana separadas, o usar la consola de desarrollador de navegador, si lo prefieres).
<!doctype html>
<html>
<head>
<meta charset="utf-8" />
<title>Consola JavaScript</title>
<style>
* {
box-sizing: border-box;
}
html {
background-color: #0c323d;
color: #809089;
font-family: monospace;
}
body {
max-width: 700px;
}
p {
margin: 0;
width: 1%;
padding: 0 1%;
font-size: 16px;
line-height: 1.5;
float: left;
}
.input p {
margin-right: 1%;
}
.output p {
width: 100%;
}
.input input {
width: 96%;
float: left;
border: none;
font-size: 16px;
line-height: 1.5;
font-family: monospace;
padding: 0;
background: #0c323d;
color: #809089;
}
div {
clear: both;
}
</style>
</head>
<body></body>
<script>
var geval = eval;
function createInput() {
var inputDiv = document.createElement("div");
var inputPara = document.createElement("p");
var inputForm = document.createElement("input");
inputDiv.setAttribute("class", "input");
inputPara.textContent = ">";
inputDiv.appendChild(inputPara);
inputDiv.appendChild(inputForm);
document.body.appendChild(inputDiv);
if (document.querySelectorAll("div").length > 1) {
inputForm.focus();
}
inputForm.addEventListener("change", executeCode);
}
function executeCode(e) {
try {
var result = geval(e.target.value);
} catch (e) {
var result = "error — " + e.message;
}
var outputDiv = document.createElement("div");
var outputPara = document.createElement("p");
outputDiv.setAttribute("class", "output");
outputPara.textContent = "Resultado: " + result;
outputDiv.appendChild(outputPara);
document.body.appendChild(outputDiv);
e.target.disabled = true;
e.target.parentNode.style.opacity = "0.5";
createInput();
}
createInput();
</script>
</html>
{{ EmbedLiveSample('Hidden_code', '100%', 300, "", "", "hide-codepen-jsfiddle") }}
Las matrices se construyen con corchetes, que contiene una lista de elementos separdos por comas.
-
Digamos que queríamos almacenar una lista de compras en un arreglo — haríamos algo como lo siguiente. Ingresa las siguientes líneas en la consola:
let shopping = ["bread", "milk", "cheese", "hummus", "noodles"]; shopping;
-
En este caso, cada elemento del arreglo es una cadena, pero ten en cuenta que puedes almacenar cualquier elemento en un arreglo — cadena, número, objeto, otra variable, incluso otro arreglo. También puedes mezclar y combinar tipos de elementos — no todos tienen que ser números, cadenas, etc. Prueba estos:
let sequence = [1, 1, 2, 3, 5, 8, 13]; let random = ["tree", 795, [0, 1, 2]];
-
Intenta creando un par de matrices por tu cuenta, antes de continuar.
Puedes entonces acceder a elementos individuales en el arreglo mediante la notación de corchetes, del mismo modo que accediste a las letras de una cadena.
-
Ingresa lo siguiente en tu consola:
shopping[0]; // devuelve "bread"
-
también puedes modificar un elemento en un arreglo simplemente dando a un item del arreglo un nuevo valor. Prueba esto:
shopping[0] = "tahini"; shopping; // shopping ahora devolverá [ "tahini", "milk", "cheese", "hummus", "noodles" ]
[!NOTE] Lo dijimos antes, pero solo como recordatorio — ¡ las computadoras comienzan a contar desde 0!
-
Ten en cuenta que un arreglo dentro de otro arreglo se llama arrelog multidimensional o matriz. Puedes acceder a los elementos de un arreglo que estén dentro de otro, encadenando dos pares de corchetes. Por ejemplo, para acceder a uno de los elementos dentro de la matriz, que a su vez, es el tercer elemento dentro de la matriz
random
(ver sección anterior), podríamos hacer algo como esto:random[2][2];
-
Intenta seguir jugando y haciendo algunas modificaciones más a tus ejemplos de matriz antes de continuar.
Puedes averiguar la longitud de un arreglo (cuántos elementos contiene) exactamente de la misma manera que determinas la longitud (en caracteres) de una cadena— utilizando la propiedad {{jsxref("Array.prototype.length","length")}}. Prueba lo siguiente:
sequence.length;
// Deve devolver 7
Esto tiene otros usos, pero se usa más comunmente para indicarle a un ciclo que continúe hasta que haya recorrido todos los elementos del arreglo. Así por ejemplo:
let sequence = [1, 1, 2, 3, 5, 8, 13];
for (var i = 0; i < sequence.length; i++) {
console.log(sequence[i]);
}
Aprenderás acerca de bucles correctamente en un artículo futuro, pero brevemente, éste código dice:
- Comienza el bucle en el elemento de la posición 0 en el arreglo.
- Detén el bucle en el número de item igual a la longitud del arreglo. Esto funcionará para un arreglo de cualquier longitid, pero en este caso el ciclo se detendrá en el elemento número 7 (esto es bueno, ya que el último elemento — que queremos que recorra el bucle — es 6.
- Para cada elemento, imprime en la consola del navegador con
console.log()
.
En esta sección veremos algunos métodos bastante útiles relacionados con matrices que nos permiten dividir cadenas en elementos de arreglo y viceversa, y agregar nuevos elementos en matrices.
A menudo se te presentarán algunos datos brutos contenidos en una cadena larga y grande, y es posible que desees separar los elementos útiles de una forma más conveniente y luego hacerle cosas, como mostrarlos en una tabla de datos. Para hacer esto, podemos usar el método {{jsxref("String.prototype.split()","split()")}}. En su forma más simple, esto toma un único parámetro, el caracter que quieres separar de la cadena, y devuelve las subcadenas entre el separador como elementos en un arreglo.
Note
Bien, esto es técnicamente un método de cadena, no un método de arreglo, pero lo hemos incluido con las matrices, ya que va bien aquí.
-
Vamos a jugar con esto, para ver como funciona. Primero, crea una cadena en tu consola:
let myData = "Manchester,London,Liverpool,Birmingham,Leeds,Carlisle";
-
Ahora dividámoslo en cada coma:
let myArray = myData.split(","); myArray;
-
Finalmente, intenta encontrar la longitud de tu nuevo arreglo y recuperar algunos elementos de ella:
myArray.length; myArray[0]; // el primer elemento del arreglo myArray[1]; // el segundo elemento del arreglo myArray[myArray.length - 1]; // el último elemento del arreglo
-
También puedes ir en la dirección opuesta usando el método {{jsxref("Array.prototype.join()","join()")}}. Prueba lo siguiente:
let myNewString = myArray.join(","); myNewString;
-
Otra forma de convertir un arreglo en cadena es usar el método {{jsxref("Array.prototype.toString()","toString()")}}.
toString()
es posiblemente más simple quejoin()
ya que no toma un parámetro, pero es más limitado. Conjoin()
puedes especificar diferentes separadores (intenta ejecutar el Paso 4 con un caracter diferente a la coma).let dogNames = ["Rocket", "Flash", "Bella", "Slugger"]; dogNames.toString(); //Rocket,Flash,Bella,Slugger
Todavia no hemos cubierto la posibilidad de agregar y eliminar elementos del arreglo — echemos un vistazo a esto ahora. Usaremos el arreglo myArray
con la que terminamos en la última sección. Si todavía no has seguido esa sección, primero crea el arreglo en tu consola:
let myArray = [
"Manchester",
"London",
"Liverpool",
"Birmingham",
"Leeds",
"Carlisle",
];
Antes que nada, para añdir o eliminar un elemento al final de un arreglo podemos usar {{jsxref("Array.prototype.push()","push()")}} y {{jsxref("Array.prototype.pop()","pop()")}} respectivamente.
-
primero usemos
push()
— nota que necesitas incluir uno o más elementos que desees agregas al final de tu arreglo. Prueba esto:myArray.push("Cardiff"); myArray; myArray.push("Bradford", "Brighton"); myArray;
-
La nueva longitud del arreglo se devuelve cuando finaliza la llamada al método. Si quisieras almacenar la nueva longitud del arreglo en una variable, podrías hacer algo como esto:
let newLength = myArray.push("Bristol"); myArray; newLength;
-
Eliminar el último elemento de un arreglo es tan simple como ejecutar
pop()
en ella. Prueba esto:myArray.pop();
-
El elemento que sé eliminó se devuelve cuando se completa la llamada al método. Para guardar este elemento en una variable, puedes hacer lo siguiente:
let removedItem = myArray.pop(); myArray; removedItem;
{{jsxref("Array.prototype.unshift()","unshift()")}} y {{jsxref("Array.prototype.shift()","shift()")}} funcionan exactamente igual de push()
y pop()
, respectivamente, excepto que funcionan al principio del arreglo, no al final.
-
Primero
unshift()
— prueba el siguiente comando:myArray.unshift("Edinburgh"); myArray;
-
Ahora
shift()
; prueba estos!let removedItem = myArray.shift(); myArray; removedItem;
Volvamos al ejemplo que describimos anteriormente — imprima los nombres de los productos y los precios en una factura, luego, sume los precios e imprímelos en la parte inferior. En el ejemplo editable a continuación, hay comentarios que contienen números — cada uno de estos marca un lugar donde debe agregar algo al código. Ellos son los siguientes:
- Debajo de
// number 1
hay un número de cadena, cada una de las cuales contiene un nombre de producto y un precio separados por dos puntos. Nos gustaría que conviertas esto en un arreglo y lo almacenamos en un arreglo llamdaproducts
. - En la misma línea que el comentario
// number 2
es el comienzo de un ciclo for. En esta línea, actualmente tenemosi <= 0
, que es una prueba condicional que hace que el bucle for se detenga inmediatamente, porque dice "detener cuandoi
es menor o igual 0", yi
comienza en 0. Nos gustaría que reemplazaras esto con una prueba condicional que detenga el ciclo cuandoi
no sea inferior a la longitud del arregloproducts
. - Justo debajo del comentario
// number 3
queremos que escriba una línea de código que divide el elemento actual del arreglo (nombre:precio
) en dos elementos separados, uno que contiene solo el nombre y otros que contienen solo el precio. Si no está seguro de cómo hacerlo, consulte el artículo Métodos de cadenas útiles para obtener ayuda o, mejor aún, consulte la sección Conversión entre cadenas y matrices de este artículo. - Como parte de la línea de código anterior, también querras convertir el precio de una cadena a un número. Si no pudes recordar como hacerlo, consulta el primer artículo de cadenas.
- Hay una variable llamada
total
que se crea y se le da un valor de 0 en la parte superior del código. Dentro del ciclo (debajo de// number 4
) queremos que agregues una línea que añade el precio actual del artículo a ese total en cada iteración del ciclo, de modo que al final del código el total correcto se imprima en la factura. Es posible que necesites un operador de asignación para hacer esto. - Queremos que cambies la línea justo de bajo
// number 5
para que la variableitemText
se iguale a "nombre de elemnto actual — $precio de elemento actual", por ejemplo "Zapatos — $23.99" en cada caso, por lo que la información correcta del artículo está impreso en la factura. Esto es simplemente una concatenación de cadenas, lo que debería ser familiar para ti.
<h2>Salida en vivo</h2>
<div class="output" style="min-height: 150px;">
<ul></ul>
<p></p>
</div>
<h2>Código editable</h2>
<p class="a11y-label">
Presione Esc para alejar el foco del área de código (Tab inserta un carácter
de tabulación).
</p>
<textarea id="code" class="playable-code" style="height: 410px;width: 95%">
var list = document.querySelector('.output ul');
var totalBox = document.querySelector('.output p');
var total = 0;
list.innerHTML = '';
totalBox.textContent = '';
// number 1
'Underpants:6.99'
'Socks:5.99'
'T-shirt:14.99'
'Trousers:31.99'
'Shoes:23.99';
for (var i = 0; i <= 0; i++) { // number 2
// number 3
// number 4
// number 5
itemText = 0;
var listItem = document.createElement('li');
listItem.textContent = itemText;
list.appendChild(listItem);
}
totalBox.textContent = 'Total: $' + total.toFixed(2);
</textarea>
<div class="playable-buttons">
<input id="reset" type="button" value="Restablecer" />
<input id="solution" type="button" value="Mostrar solución" />
</div>
var textarea = document.getElementById("code");
var reset = document.getElementById("reset");
var solution = document.getElementById("solution");
var code = textarea.value;
var userEntry = textarea.value;
function updateCode() {
eval(textarea.value);
}
reset.addEventListener("click", function () {
textarea.value = code;
userEntry = textarea.value;
solutionEntry = jsSolution;
solution.value = "Mostrar solución";
updateCode();
});
solution.addEventListener("click", function () {
if (solution.value === "Mostrar solución") {
textarea.value = solutionEntry;
solution.value = "Ocultar solución";
} else {
textarea.value = userEntry;
solution.value = "Mostrar solución";
}
updateCode();
});
var jsSolution =
"var list = document.querySelector('.output ul');\nvar totalBox = document.querySelector('.output p');\nvar total = 0;\nlist.innerHTML = '';\ntotalBox.textContent = '';\n\nvar products = ['Underpants:6.99',\n 'Socks:5.99',\n 'T-shirt:14.99',\n 'Trousers:31.99',\n 'Shoes:23.99'];\n\nfor(var i = 0; i < products.length; i++) {\n var subArray = products[i].split(':');\n var name = subArray[0];\n var price = Number(subArray[1]);\n total += price;\n itemText = name + ' — $' + price;\n\n var listItem = document.createElement('li');\n listItem.textContent = itemText;\n list.appendChild(listItem);\n}\n\ntotalBox.textContent = 'Total: $' + total.toFixed(2);";
var solutionEntry = jsSolution;
textarea.addEventListener("input", updateCode);
window.addEventListener("load", updateCode);
// detener la tecla de tabulación fuera del área de texto y
// hacer que escriba una tabulación en la posición del cursor
textarea.onkeydown = function (e) {
if (e.keyCode === 9) {
e.preventDefault();
insertAtCaret("\t");
}
if (e.keyCode === 27) {
textarea.blur();
}
};
function insertAtCaret(text) {
var scrollPos = textarea.scrollTop;
var caretPos = textarea.selectionStart;
var front = textarea.value.substring(0, caretPos);
var back = textarea.value.substring(
textarea.selectionEnd,
textarea.value.length,
);
textarea.value = front + text + back;
caretPos = caretPos + text.length;
textarea.selectionStart = caretPos;
textarea.selectionEnd = caretPos;
textarea.focus();
textarea.scrollTop = scrollPos;
}
// Actualice el código de usuario guardado cada vez que el usuario actualice el código de área de texto
textarea.onkeyup = function () {
// Solo queremos guardar el estado cuando se muestra el código de usuario,
// no la solución, para que esa solución no se guarde sobre el código de usuario.
if (solution.value === "Mostrar solución") {
userEntry = textarea.value;
} else {
solutionEntry = textarea.value;
}
updateCode();
};
html {
font-family: sans-serif;
}
h2 {
font-size: 16px;
}
.a11y-label {
margin: 0;
text-align: right;
font-size: 0.7rem;
width: 98%;
}
body {
margin: 10px;
background-color: #f5f9fa;
}
{{ EmbedLiveSample('Playable_code', '100%', 730, "", "", "hide-codepen-jsfiddle") }}
Un buen uso para los métodos de arreglo como {{jsxref("Array.prototype.push()","push()")}} y {{jsxref("Array.prototype.pop()","pop()")}} es cuando estás manteniendo un registro de elementos actualmente activos en una aplicación web. En una escena animada por ejemplo, es posible que tengas un arreglo de objetos que representan los gráficos de fondo que se muestran actualmente, y es posible que sólo desees que se muestren 50 a la vez, por razones de rendimiento o desorden. A medida que se crean y agregan nuevos objetos al arreglo, se puede eliminar los más antiguos del arreglo para mantener el número deseado.
En este ejemplo vamos a mostrar un uso mucho más simple — aquí te daremos un sitio de búsqueda falso, con un cuadro de búsqueda. La idea es que cuando los términos se ingresan en un cuadro de búsqueda, se muetren el top 5 de términos de búsqueda previos en la lista. Cuando el número de términos supera el 5, el último término comienza a borrarse cada vez que agregas un nuevo término a la parte superior, por lo que siempre se muestran los 5 términos anteriores.
Note
En una aplicación de búsqueda real, probablemente puedas hacer clic en los términos de búsqueda anteriores para volver a los términos de búsqueda anteriores y ¡se motrarán los resultados de búsqueda reales! Solamente lo mantendremos simple por ahora.
Para completar la aplicación necesitamos:
- Agregar una línea debajo del comentario
// number 1
que agrega el valor actual ingresado en la entrada de la búsqueda al inicio del arreglo. Esto se puede recuperar usandosearchInput.value
. - Agrega una línea debajo del comentario
// number 2
que elimina el valor actualmente al final del arreglo.
<h2>Salida en vivo</h2>
<div class="output" style="min-height: 150px;">
<input type="text" /><button>Buscar</button>
<ul></ul>
</div>
<h2>Código editable</h2>
<p class="a11y-label">
Presione Esc para alejar el foco del área de código (Tab inserta un carácter
de tabulación).
</p>
<textarea id="code" class="playable-code" style="height: 370px; width: 95%">
var list = document.querySelector('.output ul');
var searchInput = document.querySelector('.output input');
var searchBtn = document.querySelector('.output button');
list.innerHTML = '';
var myHistory = [];
searchBtn.onclick = function() {
// Solo permitiremos que se ingrese un término si la entrada de búsqueda no está vacía.
if (searchInput.value !== '') {
// number 1
// Vacíe la lista para no mostrar entradas duplicadas. La pantalla
// se regenera cada vez que se ingresa un término de búsqueda.
list.innerHTML = '';
// recorrer el arreglo y mostrar todos los términos de búsqueda en la lista
for (var i = 0; i < myHistory.length; i++) {
itemText = myHistory[i];
var listItem = document.createElement('li');
listItem.textContent = itemText;
list.appendChild(listItem);
}
// Si la longitud del arreglo es 5 o más, elimine el término de búsqueda más antiguo
if (myHistory.length >= 5) {
// number 2
}
// vacíe la entrada de búsqueda y enfóquela, listo para ingresar el siguiente término
searchInput.value = '';
searchInput.focus();
}
}
</textarea>
<div class="playable-buttons">
<input id="reset" type="button" value="Restablecer" />
<input id="solution" type="button" value="Mostrar solución" />
</div>
html {
font-family: sans-serif;
}
h2 {
font-size: 16px;
}
.a11y-label {
margin: 0;
text-align: right;
font-size: 0.7rem;
width: 98%;
}
body {
margin: 10px;
background: #f5f9fa;
}
var textarea = document.getElementById("code");
var reset = document.getElementById("reset");
var solution = document.getElementById("solution");
var code = textarea.value;
var userEntry = textarea.value;
function updateCode() {
eval(textarea.value);
}
reset.addEventListener("click", function () {
textarea.value = code;
userEntry = textarea.value;
solutionEntry = jsSolution;
solution.value = "Mostrar solución";
updateCode();
});
solution.addEventListener("click", function () {
if (solution.value === "Mostrar solución") {
textarea.value = solutionEntry;
solution.value = "Ocultar solución";
} else {
textarea.value = userEntry;
solution.value = "Mostrar solución";
}
updateCode();
});
var jsSolution =
"var list = document.querySelector('.output ul');\nvar searchInput = document.querySelector('.output input');\nvar searchBtn = document.querySelector('.output button');\n\nlist.innerHTML = '';\n\nvar myHistory= [];\n\nsearchBtn.onclick = function() {\n if(searchInput.value !== '') {\n myHistory.unshift(searchInput.value);\n\n list.innerHTML = '';\n\n for(var i = 0; i < myHistory.length; i++) {\n itemText = myHistory[i];\n var listItem = document.createElement('li');\n listItem.textContent = itemText;\n list.appendChild(listItem);\n }\n\n if(myHistory.length >= 5) {\n myHistory.pop();\n }\n\n searchInput.value = '';\n searchInput.focus();\n }\n}";
var solutionEntry = jsSolution;
textarea.addEventListener("input", updateCode);
window.addEventListener("load", updateCode);
// detener la tecla de tabulación fuera del área de texto y
// hacer que escriba una tabulación en la posición del cursor
textarea.onkeydown = function (e) {
if (e.keyCode === 9) {
e.preventDefault();
insertAtCaret("\t");
}
if (e.keyCode === 27) {
textarea.blur();
}
};
function insertAtCaret(text) {
var scrollPos = textarea.scrollTop;
var caretPos = textarea.selectionStart;
var front = textarea.value.substring(0, caretPos);
var back = textarea.value.substring(
textarea.selectionEnd,
textarea.value.length,
);
textarea.value = front + text + back;
caretPos = caretPos + text.length;
textarea.selectionStart = caretPos;
textarea.selectionEnd = caretPos;
textarea.focus();
textarea.scrollTop = scrollPos;
}
// Actualice el código de usuario guardado cada vez que el usuario actualice el código de área de texto
textarea.onkeyup = function () {
// Solo queremos guardar el estado cuando se muestra el código de usuario,
// no la solución, para que esa solución no se guarde sobre el código de usuario.
if (solution.value === "Mostrar solución") {
userEntry = textarea.value;
} else {
solutionEntry = textarea.value;
}
updateCode();
};
{{ EmbedLiveSample('Playable_code_2', '100%', 700, "", "", "hide-codepen-jsfiddle") }}
Después de leer este artículo, estamos seguros de que estaras de acuerdo en que las matrices parecen bastante útiles; las verás aparecer en todas partes en JavaScript, a menudo en asociación con bucles para hacer una misma cosa con cada elemento del arreglo. Te enseñaremos todos los aspectos básicos útiles que hay que conocer sobre los bucles en el siguiente módulo, pero por ahora debes darte un aplauso y tomarte un merecido descanso; ¡has trabajado en todos los artículos de este módulo!
Lo único que queda por hacer es trabajar a través de la evaluación de este módulo, que te pondrá a prueba tu comprensión de los de los artículos anteriores.
- Colecciones indexadas — una guía de nivel avanzado sobre arreglos y sus primos, los arreglos tipadosa.
- {{jsxref("Array")}}: la página de referencia del objeto
Array
, para obtener una guía de referencia detallada de las funciones analizadas en esta página y muchas más.
{{PreviousMenuNext("Learn/JavaScript/First_steps/Useful_string_methods", "Learn/JavaScript/First_steps/Silly_story_generator", "Learn/JavaScript/First_steps")}}