Esta página ha sido traducida del inglés por la comunidad.Aprende más y únete a la comunidad de MDN Web Docs.
Código de bucle
Los lenguajes de programación son muy útiles para completar rápidamente tareas repetitivas, desde múltiples cálculos básicos hasta casi cualquier otra situación en la que tenga que completar muchos elementos de trabajo similares. Aquí veremos las estructuras de bucle disponibles en JavaScript que manejan tales necesidades.
| Prerrequisitos: | Conocimientos básicos de informática, una comprensión básica de HTML y CSS, Primeros pasos de JavaScript. |
|---|---|
| Objetivo: | Comprender cómo usar bucles en JavaScript. |
In this article
- ¿Por qué son útiles los bucles?
- Recorriendo una colección
- El bucle estándar for
- Saliendo de bucles con break
- Omitir iteraciones con continue
- while y do...while
- Aprendizaje activo: iniciar cuenta regresiva
- Aprendizaje activo: rellenar una lista de invitados
- ¿Qué tipo de bucle debes usar?
- Pon a prueba tus habilidades
- Conclusión
- Vease también
¿Por qué son útiles los bucles?
Los bucles se tratan de hacer lo mismo una y otra vez. A menudo, el código será ligeramente diferente cada vez que dure el bucle, o se ejecutará el mismo código pero con diferentes variables.
Ejemplo de código de bucles
Supongamos que queremos dibujar 100 círculos aleatorios en un elemento<canvas> (pulse el botónActualizar para ejecutar el ejemplo una y otra vez para ver diferentes conjuntos aleatorios):
<button>Actualizar</button> <canvas></canvas>html { width: 100%; height: inherit; background: #ddd;}canvas { display: block;}body { margin: 0;}button { position: absolute; top: 5px; left: 5px;}Este es el código JavaScript que implementa este ejemplo:
const btn = document.querySelector("button");const canvas = document.querySelector("canvas");const ctx = canvas.getContext("2d");document.addEventListener("DOMContentLoaded", () => { canvas.width = document.documentElement.clientWidth; canvas.height = document.documentElement.clientHeight;});function random(number) { return Math.floor(Math.random() * number);}function draw() { ctx.clearRect(0, 0, canvas.width, canvas.height); for (let i = 0; i < 100; i++) { ctx.beginPath(); ctx.fillStyle = "rgba(255,0,0,0.5)"; ctx.arc( random(canvas.width), random(canvas.height), random(50), 0, 2 * Math.PI, ); ctx.fill(); }}btn.addEventListener("click", draw);Con y sin bucles
No tienes que entender todo el código por ahora, pero veamos la parte del código que realmente dibuja los 100 círculos:
for (let i = 0; i < 100; i++) { ctx.beginPath(); ctx.fillStyle = "rgba(255,0,0,0.5)"; ctx.arc( random(canvas.width), random(canvas.height), random(50), 0, 2 * Math.PI, ); ctx.fill();}random(x), definido anteriormente en el código, devuelve un número entero entre0yx-1.
Deberías tener la idea básica: estamos usando un bucle para ejecutar 100 iteraciones de este código, cada una de las cuales dibuja un círculo en una posición aleatoria en la página.La cantidad de código necesaria sería la misma si estuviéramos dibujando 100 círculos, 1000 o 10.000.Solo un número tiene que cambiar.
Si no estuviéramos usando un bucle aquí, tendríamos que repetir el siguiente código para cada círculo que quisiéramos dibujar:
ctx.beginPath();ctx.fillStyle = "rgba(255,0,0,0.5)";ctx.arc( random(canvas.width), random(canvas.height), random(50), 0, 2 * Math.PI,);ctx.fill();Esto sería muy aburrido y difícil de mantener.
Recorriendo una colección
La mayoría de las veces, cuando usas un bucle, tendrás una colección de artículos y querrás hacer algo con cada artículo.
Un tipo de colección es elArreglo, que conocimos en el capítulo deArreglos de este curso.Pero también hay otras colecciones en JavaScript, comoSet yMap.
El bucle for...of
La herramienta básica para recorrer una colección es el buclefor...of:
const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];for (const cat of cats) { console.log(cat);}En este ejemplo,for (const cat of cats) dice:
- Dada la colección
cats, consigue el primer artículo de la colección. - Asignarlo a la variable
caty luego ejecutar el código entre las llaves{}. - Obtén el siguiente elemento y repite (2) hasta que hayas llegado al final de la colección.
map() y filter()
JavaScript también tiene bucles más especializados para colecciones, y mencionaremos dos de ellos aquí.
Puede usarmap() para hacer algo con cada elemento de una colección y crear una nueva colección que contenga los elementos modificados:
function toUpper(string) { return string.toUpperCase();}const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];const upperCats = cats.map(toUpper);console.log(upperCats);// [ "LEOPARDO", "SERVAL", "JAGUAR", "TIGRE", "CARACAL", "LEÓN" ]Aquí pasamos una función acats.map(), ymap() llama a la función una vez por cada elemento de la matriz, pasando el elemento. A continuación, añade el valor devuelto de cada llamada de función a un nuevo arreglo y, finalmente, devuelve el nuevo arreglo. En este caso, la función que proporcionamos convierte el elemento en mayúsculas, por lo que la matriz resultante contiene todos nuestros gatos en mayúsculas:
[ "LEOPARDO", "SERVAL", "JAGUAR", "TIGRE", "CARACAL", "LEÓN" ]Puede usarfilter() para probar cada elemento de una colección y crear una nueva colección que contenga solo elementos que coincidan:
function lCat(cat) { return cat.startsWith("L");}const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];const filtered = cats.filter(lCat);console.log(filtrado);// [ "Leopardo", "León" ]Esto se parece mucho amap(), excepto que la función que pasamos devuelve unbooleano: si devuelvetrue, entonces el elemento se incluye en el nuevo arreglo.Nuestra función prueba que el elemento comienza con la letra "L", por lo que el resultado es una matriz que contiene solo gatos cuyos nombres comienzan con "L":
[ "Leopardo", "León" ]Tenga en cuenta quemap() yfilter() se usan a menudo conexpresiones de funciones, que aprenderemos en el móduloFunctions.Usando expresiones de función podríamos reescribir el ejemplo anterior para que sea mucho más compacto:
const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];const filter = cats.filter((cat) => cat.startsWith("L"));console.log(filtrado);// [ "Leopardo", "León" ]El bucle estándar for
En el ejemplo anterior de "círculos de dibujo", no tiene una colección de elementos para recorrer: realmente solo desea ejecutar el mismo código 100 veces.En un caso como ese, debes usar el buclefor.Tiene la siguiente sintaxis:
for (inicializador; condición; expresión-final) { // código a ejecutar}Aquí tenemos:
La palabra clave
for, seguida de algunos paréntesis.Dentro de los paréntesis tenemos tres ítems, separados por punto y coma:
- Uninicializador: generalmente es una variable establecida en un número, que se incrementa para contar el número de veces que se ha ejecutado el bucle.También se denomina a vecesvariable de contador.
- Unacondición: define cuándo el bucle debe dejar de funcionar.Esta es generalmente una expresión que presenta un operador de comparación, una prueba para ver si se ha cumplido la condición de salida.
- Unaexpresión-final: siempre se evalúa (o ejecuta) cada vez que el bucle ha pasado por una iteración completa.Por lo general, sirve para incrementar (o en algunos casos disminuir) la variable contadora, para acercarla al punto en que la condición ya no es
true.
Algunas llaves que contienen un bloque de código: este código se ejecutará cada vez que el bucle se repita.
Cálculo de cuadrados
Veamos un ejemplo real para que podamos visualizar lo que estos hacen con mayor claridad.
<button>Calcular</button><button>Borrar</button><pre></pre>const results = document.querySelector("#results");function calculate() { for (let i = 1; i < 10; i++) { const newResult = `${i} x ${i} = ${i * i}`; results.textContent += `${newResult}\n`; } results.textContent += "\n¡Finalizado!";}const calculateBtn = document.querySelector("#calculate");const clearBtn = document.querySelector("#clear");calculateBtn.addEventListener("click", calculate);clearBtn.addEventListener("click", () => (results.textContent = ""));Esto nos da el siguiente resultado:
Este código calcula los cuadrados de los números del 1 al 9 y escribe el resultado. El núcleo del código es el buclefor que realiza el cálculo.
Desglosemos la líneafor (let i = 1; i < 10; i++) en sus tres partes:
let i = 1: la variable del contador,i, comienza en1. Tenga en cuenta que tenemos que usarletpara el contador, porque lo estamos reasignando cada vez que damos la vuelta al bucle.i < 10: sigue dando la vuelta al bucle mientrasisea menor que10.i++: añade uno aicada vez que recorras el bucle.
Dentro del bucle, calculamos el cuadrado del valor actual dei, es decir:i * i. Creamos una cadena que expresa el cálculo que realizamos y el resultado, y añadimos esta cadena al texto de salida. También añadimos\n, por lo que la siguiente cadena que añadamos comenzará en una nueva línea. De manera que:
- Durante la primera ejecución,
i = 1, por lo que añadiremos1 x 1 = 1. - Durante la segunda ejecución,
i = 2, por lo que añadiremos2 x 2 = 4. - Y así sucesivamente...
- Cuando
isea igual a10, dejaremos de ejecutar el bucle y pasaremos directamente al siguiente código debajo del bucle, imprimiendo el mensaje¡Finalizado!en una nueva línea.
Recorriendo colecciones con un bucle for
Puede usar un buclefor para iterar a través de una colección, en lugar de un buclefor...of.
Echemos un vistazo de nuevo a nuestro ejemplo anterior "for...of":
const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];for (const cat of cats) { console.log(cat);}Podríamos reescribir ese código así:
const cats = ["Leopardo", "Serval", "Jaguar", "Tigre", "Caracal", "León"];for (let i = 0; i < cats.length; i++) { console.log(cats[i]);}En este bucle, comenzamosi en0 y nos detenemos cuandoi alcanza la longitud del arreglo.Luego, dentro del bucle, estamos usandoi para acceder a cada elemento del arreglo a su vez.
Esto funciona muy bien, y en las primeras versiones de JavaScript,for...of no existía, por lo que esta era la forma estándar de iterar a través de un arreglo.Sin embargo, ofrece más posibilidades de introducir errores en tu código. Por ejemplo:
- puede comenzar
ien1, olvidando que el primer índice del arreglo es cero, no 1. - puede detenerse en
i <= cats.length, olvidando que el último índice de matriz está enlength - 1.
Por razones como esta, generalmente es mejor usarfor...of si puedes.
A veces todavía necesitas usar un buclefor para iterar a través de un arreglo.Por ejemplo, en el siguiente código queremos registrar un mensaje que enumere a nuestros gatos:
const cats = ["Pete", "Biggles", "Jasmine"];let myFavoriteCats = "Mis gatos se llaman ";for (const cat of cats) { myFavoriteCats += `${cat}, `;}console.log(myFavoriteCats); // "Mis gatos se llaman Pete, Biggles, Jasmine, "La oración de salida final no está muy bien formada:
Mis gatos se llaman Pete, Biggles, Jasmine,
Preferiríamos que manejara al último gato de manera diferente, así:
Mis gatos se llaman Pete, Biggles y Jasmine.
Pero para hacer esto necesitamos saber cuándo estamos en la iteración final del bucle, y para hacerlo podemos usar un buclefor y examinar el valor dei:
const cats = ["Pete", "Biggles", "Jasmine"];let myFavoriteCats = "Mis gatos se llaman ";for (let i = 0; i < cats.length; i++) { if (i === cats.length - 1) { // Estamos al final del arreglo myFavoriteCats += `y ${cats[i]}.`; } else { myFavoriteCats += `${cats[i]}, `; }}console.log(myFavoriteCats); // "Mis gatos se llaman Pete, Biggles y Jasmine."Saliendo de bucles con break
Si desea salir de un bucle antes de que se hayan completado todas las iteraciones, puede usar la instrucciónbreak.Ya vimos esto en el artículo anterior cuando analizamos lassentencias switch: cuando se cumple un caso en una sentencia switch que coincide con la expresión de entrada, la sentenciabreak sale inmediatamente de la sentencia switch y pasa al código después de ella.
Es lo mismo con los bucles: una instrucciónbreak saldrá inmediatamente del bucle y hará que el navegador pase a cualquier código que lo siga.
Digamos que queríamos buscar a través de una serie de contactos y números de teléfono y devolver solo el número que queríamos encontrar.Primero, un HTML simple: un<input> de texto que nos permite ingresar un nombre para buscar, un elemento<button> para enviar una búsqueda y un elemento<p> para mostrar los resultados en:
<label for="search">Buscar por nombre de contacto: </label><input type="text" /><button>Buscar</button><p></p>Ahora pasemos a #"Chris:2232322", "Sarah:3453456", "Bill:7654322", "Mary:9998769", "Dianne:9384975",];const para = document.querySelector("p");const input = document.querySelector("input");const btn = document.querySelector("button");btn.addEventListener("click", () => { const searchName = input.value.toLowerCase(); input.value = ""; input.focus(); para.textContent = ""; for (const contact of contacts) { const splitContact = contact.split(":"); if (splitContact[0].toLowerCase() === searchName) { para.textContent = `El número de ${splitContact[0]} es ${splitContact[1]}.`; break; } } if (para.textContent === "") { para.textContent = "Contacto no encontrado."; }});
En primer lugar, tenemos algunas definiciones de variables: tenemos una variedad de información de contacto, y cada elemento es una cadena que contiene un nombre y un número de teléfono separados por dos puntos.
A continuación, adjuntamos un detector de eventos al botón (
btn) para que cuando se pulse se ejecute algún código para realizar la búsqueda y devolver los resultados.Almacenamos el valor introducido en la entrada de texto en una variable llamada
searchName, antes de vaciar la entrada de texto y volver a enfocarla, listos para la siguiente búsqueda.Tenga en cuenta que también ejecutamos el métodotoLowerCase()en la cadena, de modo que las búsquedas no distingan entre mayúsculas y minúsculas.Ahora pasemos a la parte interesante, el bucle
for...of:- Dentro del bucle, primero dividimos el contacto actual en el carácter de dos puntos y almacenamos los dos valores resultantes en un arreglo llamado
splitContact. - Luego usamos una instrucción condicional para probar si
splitContact [0](el nombre del contacto, nuevamente en minúsculas contoLowerCase()) es igual alsearchNameingresado.Si es así, introducimos una cadena en el párrafo para informar cuál es el número del contacto y usamosbreakpara finalizar el bucle.
- Dentro del bucle, primero dividimos el contacto actual en el carácter de dos puntos y almacenamos los dos valores resultantes en un arreglo llamado
Después del bucle, verificamos si configuramos un contacto y, de lo contrario, configuramos el texto del párrafo como "Contacto no encontrado".
Nota:También puedes ver elcódigo fuente completo en GitHub (tambiénverlo en vivo).
Omitir iteraciones con continue
La instruccióncontinue funciona de manera similar abreak, pero en lugar de salir del bucle por completo, salta a la siguiente iteración del bucle.Veamos otro ejemplo que toma un número como entrada y devuelve solo los números que son cuadrados de enteros (números enteros).
El HTML es básicamente el mismo que el último ejemplo: una entrada numérica simple y un párrafo para la salida.
<label for="number">Introducir número: </label><input type="number" /><button>Generar cuadrados enteros</button><p>Resultado:</p>El JavaScript también es casi el mismo, aunque el bucle en sí es un poco diferente:
const para = document.querySelector("p");const input = document.querySelector("input");const btn = document.querySelector("button");btn.addEventListener("click", () => { para.textContent = "Resultado: "; const num = input.value; input.value = ""; input.focus(); for (let i = 1; i <= num; i++) { let sqRoot = Math.sqrt(i); if (Math.floor(sqRoot) !== sqRoot) { continue; } para.textContent += `${i} `; }});Este es el resultado:
- En este caso, la entrada debe ser un número (
num). Al bucleforse le da un contador que comienza en 1 (ya que no estamos interesados en 0 en este caso), una condición de salida que dice que el bucle se detendrá cuando el contador se vuelva más grande que la entradanum, y un iterador que suma 1 al contador cada vez. - Dentro del bucle, encontramos la raíz cuadrada de cada número usandoMath.sqrt(i), luego verificamos si la raíz cuadrada es un entero probando si es igual a sí misma cuando se ha redondeado al entero más cercano (esto es lo queMath.floor() hace al número que se pasa).
- Si la raíz cuadrada y la raíz cuadrada redondeada hacia abajo no son iguales entre sí (
!==), significa que la raíz cuadrada no es un número entero, por lo que no nos interesa. En tal caso, usamos la instruccióncontinuepara saltar a la siguiente iteración de bucle sin registrar el número en ninguna parte. - Si la raíz cuadrada es un número entero, omitimos por completo el bloque
if, por lo que no se ejecuta la instruccióncontinue; en su lugar, concatenamos el valoriactual más un espacio al final del contenido del párrafo.
Nota:También puedes ver elcódigo fuente completo en GitHub (tambiénverlo en vivo).
while y do...while
for no es el único tipo de bucle disponible en JavaScript. En realidad, hay muchos otros y, aunque no es necesario que entiendas todos estos ahora, vale la pena echar un vistazo a la estructura de un par de otros para que puedas reconocer las mismas características en el trabajo de una manera ligeramente diferente.
Primero, echemos un vistazo al buclewhile. La sintaxis de este bucle se ve así:
inicializadorwhile(condición) { // código a ejecutar expresión-final}Esto funciona de una manera muy similar al buclefor, excepto que la variable inicializadora se establece antes del bucle, y la expresión final se incluye dentro del bucle después del código a ejecutar, en lugar de que estos dos elementos se incluyan dentro de los paréntesis.La condición se incluye dentro de los paréntesis, que están precedidos por la palabra clavewhile en lugar defor.
Los mismos tres elementos todavía están presentes, y todavía están definidos en el mismo orden en que están en el bucle for.Esto se debe a que debe tener un inicializador definido antes de poder verificar si la condición es verdadera o no.La expresión final se ejecuta después de que se haya ejecutado el código dentro del bucle (se ha completado una iteración), lo que solo ocurrirá si la condición sigue siendo cierta.
Echemos un vistazo de nuevo a nuestro ejemplo de lista de gatos, pero reescrito para usar un bucle while:
const cats = ["Pete", "Biggles", "Jasmine"];let myFavoriteCats = "Mis gatos se llaman ";let i = 0;while (i < cats.length) { if (i === cats.length - 1) { myFavoriteCats += `y ${cats[i]}.`; } else { myFavoriteCats += `${cats[i]}, `; } i++;}console.log(myFavoriteCats); // "Mis gatos se llaman Pete, Biggles y Jasmine."Nota:Esto sigue funcionando exactamente como se esperaba: échale un vistazo alcódigo fuente completo (también puedes ver elcódigo fuente completo).
El bucledo...while es muy similar, pero proporciona una variación en la estructura while:
inicializadordo { // código a ejecutar expresión-final} while (condición)En este caso, el inicializador vuelve a aparecer primero, antes de que comience el bucle. La palabra clave precede directamente a las llaves que contienen el código a ejecutar y la expresión final.
La principal diferencia entre un bucledo...while y un buclewhile es queel código dentro de un bucledo...while siempre se ejecuta al menos una vez. Esto se debe a que la condición viene después del código dentro del bucle. Así que siempre ejecutamos ese código, luego verificamos si necesitamos ejecutarlo de nuevo. En los bucleswhile yfor, la comprobación es lo primero, por lo que es posible que el código nunca se ejecute.
Volvamos a escribir nuestro ejemplo de listado de gatos para usar un bucle de "do...while":
const cats = ["Pete", "Biggles", "Jasmine"];let myFavoriteCats = "Mis gatos se llaman ";let i = 0;do { if (i === cats.length - 1) { myFavoriteCats += `y ${cats[i]}.`; } else { myFavoriteCats += `${cats[i]}, `; } i++;} while (i < cats.length);console.log(myFavoriteCats); // "Mis gatos se llaman Pete, Biggles y Jasmine."Nota:De nuevo, esto funciona igual que lo esperado: échale un vistazo alcódigo fuente completo (también puedes ver elcódigo fuente completo).
Advertencia:Con while y do...while, como con todos los bucles, debe asegurarse de que el inicializador se incremente o, según el caso, se disminuya, para que la condición finalmente se vuelva falsa.Si no, el bucle continuará para siempre y el navegador lo obligará a detenerse o se bloqueará. Esto se llama unbucle infinito.
Aprendizaje activo: iniciar cuenta regresiva
En este ejercicio, queremos que imprima una cuenta regresiva de lanzamiento simple para la caja de salida, desde 10 hasta Blastoff.En concreto, queremos:
Bucle de 10 a 0. Te hemos proporcionado un inicializador:
let i = 10;.Para cada iteración, cree un nuevo párrafo y añádalo a la salida
<div>, que hemos seleccionado usandoconst output = document.querySelector('.output');.En los comentarios, le proporcionamos tres líneas de código que deben usarse en algún lugar dentro del bucle:const para = document.createElement('p');: crea un nuevo párrafo.output.appendChild(para);: añade el párrafo a la salida<div>.para.textContent =: hace que el texto dentro del párrafo sea igual a lo que pongas en el lado derecho, después del signo igual.
Los diferentes números de iteración requieren que se coloque un texto diferente en el párrafo para esa iteración (necesitarás una instrucción condicional y varias líneas de
para.textContent =):- Si el número es 10, imprima "Cuenta regresiva 10" en el párrafo.
- Si el número es 0, imprima "Blast off!" en el párrafo.
- Para cualquier otro número, imprime solo el número en el párrafo.
¡Recuerda incluir un iterador! Sin embargo, en este ejemplo estamos contando hacia atrás después de cada iteración, no hacia arriba, por lo queno quieres
i++: ¿cómo iteras hacia abajo?
Nota:Si comienza a escribir el bucle (por ejemplo, (while(i>=0)), es posible que el navegador se atasque porque aún no ha ingresado la condición final. Así que ten cuidado con esto. Puedes empezar a escribir tu código en un comentario para hacer frente a este problema y eliminar el comentario después de terminar.
Si comete un error, siempre puede restablecer el ejemplo con el botón "Restablecer".Si te quedas realmente atascado, pulsa "Mostrar solución" para ver una solución.
<h2>Salida en vivo</h2><div></div><h2>Código editable</h2><p> Pulse Esc para alejar el foco del área de código (Tab inserta un carácter de tabulación).</p><textarea>let output = document.querySelector('.output');output.innerHTML = '';// let i = 10;// const para = document.createElement('p');// para.textContent = ;// output.appendChild(para);</textarea><div> <input type="button" value="Restablecer" /> <input 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;}const textarea = document.getElementById("code");const reset = document.getElementById("reset");const solution = document.getElementById("solution");let code = textarea.value;let 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();});let jsSolution = `const output = document.querySelector('.output');output.innerHTML = '';let i = 10;while (i >= 0) { const para = document.createElement('p'); if (i === 10) { paragraphtextContent = \`Cuenta atrás \${i}\`; } else if (i === 0) { para.textContent = 'Blast off!'; } else { para.textContent = i; } output.appendChild(para); i--;}`;let solutionEntry = jsSolution;textarea.addEventListener("input", updateCode);window.addEventListener("load", updateCode);// detener la tabulación de la tecla tabulación fuera del área de texto y// hacer que escriba una tabulación en la posición del cursor en su lugartextarea.onkeydown = function (e) { if (e.keyCode === 9) { e.preventDefault(); insertAtCaret("\t"); } if (e.keyCode === 27) { textarea.blur(); }};function insertAtCaret(text) { const scrollPos = textarea.scrollTop; let caretPos = textarea.selectionStart; const front = textarea.value.substring(0, caretPos); const back = textarea.value.substring( textarea.selectionEnd, textarea.value.length, ); textarea.value = front + text + back; caretPos += text.length; textarea.selectionStart = caretPos; textarea.selectionEnd = caretPos; textarea.focus(); textarea.scrollTop = scrollPos;}// Actualizar el userCode guardado cada vez que el usuario actualice el código de área de textotextarea.onkeyup = () => { // Solo queremos guardar el estado cuando se muestra el código de usuario, // no la solución, para que la 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();};Aprendizaje activo: rellenar una lista de invitados
En este ejercicio, queremos que tomes una lista de nombres almacenados en un arreglo y los pongas en una lista de invitados. Pero no es tan fácil: ¡no queremos dejar entrar a Phil y Lola porque son codiciosos y groseros, y siempre comen toda la comida! Tenemos dos listas, una para que los huéspedes la admitan y otra para que los huéspedes la rechacen.
En concreto, queremos:
- Escribe un bucle que itere a través del arreglo
people. - Durante cada iteración de bucle, compruebe si el elemento del arreglo actual es igual a "Phil" o "Lola" utilizando una instrucción condicional:
- Si es así, concatene el elemento del arreglo al final de
textContentdel párraforefused, seguido de una coma y un espacio. - Si no es así, concatene el elemento del arreglo hasta el final del
textContentdel párrafoadmitted, seguido de una coma y un espacio.
- Si es así, concatene el elemento del arreglo al final de
Ya te hemos proporcionado:
refused.textContent +=: los inicios de una línea que concatenará algo al final derefused.textContent.admitted.textContent +=; los inicios de una línea que concatenará algo al final deadmitted.textContent.
Pregunta de bonificación adicional: después de completar las tareas anteriores con éxito, te quedarán dos listas de nombres, separadas por comas, pero estarán desordenadas: habrá una coma al final de cada una.¿Puedes averiguar cómo escribir líneas que corten la última coma en cada caso y añadir un punto al final?Consulta el artículoMétodos de cadenas útiles para obtener ayuda.
Si comete un error, siempre puede restablecer el ejemplo con el botón "Restablecer".Si te quedas realmente atascado, pulsa "Mostrar solución" para ver una solución.
<h2>Salida en vivo</h2><div> <p>Admitir:</p> <p>Rechazar:</p></div><h2>Código editable</h2><p> Pulse Esc para alejar el foco del área de código (Tab inserta un carácter de tabulación).</p><textarea>const people = ['Chris', 'Anne', 'Colin', 'Terri', 'Phil', 'Lola', 'Sam', 'Kay', 'Bruce'];const admitted = document.querySelector('.admitted');const refused = document.querySelector('.refused');admitted.textContent = 'Admitir: ';refused.textContent = 'Rechazar: ';// El bucle comienza aquí// refused.textContent += ;// admitted.textContent += ;</textarea><div> <input type="button" value="Restablecer" /> <input 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;}const textarea = document.getElementById("code");const reset = document.getElementById("reset");const solution = document.getElementById("solution");let code = textarea.value;let 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();});const jsSolution = `const people = ['Chris', 'Anne', 'Colin', 'Terri', 'Phil', 'Lola', 'Sam', 'Kay', 'Bruce'];const admitted = document.querySelector('.admitted');const refused = document.querySelector('.refused');admitted.textContent = 'Admitir: ';refused.textContent = 'Rechazar: ';for (const person of people) { if (person === 'Phil' || person === 'Lola') { refused.textContent += \`\${person}, \`; } else { admitted.textContent += \`\${person}, \`; }}refused.textContent = refused.textContent.slice(0,refused.textContent.length-2) + '.';admitted.textContent = admitted.textContent.slice(0,admitted.textContent.length-2) + '.';`;let solutionEntry = jsSolution;textarea.addEventListener("input", updateCode);window.addEventListener("load", updateCode);// detener la tabulación de la tecla tabulación fuera del área de texto y// hacer que escriba una tabulación en la posición del cursor en su lugartextarea.onkeydown = function (e) { if (e.keyCode === 9) { e.preventDefault(); insertAtCaret("\t"); } if (e.keyCode === 27) { textarea.blur(); }};function insertAtCaret(text) { const scrollPos = textarea.scrollTop; let caretPos = textarea.selectionStart; const front = textarea.value.substring(0, caretPos); const back = textarea.value.substring( textarea.selectionEnd, textarea.value.length, ); textarea.value = front + text + back; caretPos += text.length; textarea.selectionStart = caretPos; textarea.selectionEnd = caretPos; textarea.focus(); textarea.scrollTop = scrollPos;}// Actualizar el userCode guardado cada vez que el usuario actualice el código de área de textotextarea.onkeyup = () => { // Solo queremos guardar el estado cuando se muestra el código de usuario, // no la solución, para que la 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();};¿Qué tipo de bucle debes usar?
Si está iterando a través de un arreglo o algún otro objeto que lo admita, y no necesita acceder a la posición de índice de cada elemento, entoncesfor...of es la mejor opción. Es más fácil de leer y hay menos para equivocarse.
Para otros usos, los buclesfor,while ydo...while son en gran medida intercambiables.Todos se pueden usar para resolver los mismos problemas, y cuál uses dependerá en gran medida de tus preferencias personales: cuál te resulta más fácil de recordar o más intuitivo.Recomendaríamosfor, al menos para empezar, ya que es probablemente el más fácil para recordar todo: el inicializador, la condición y la expresión final tienen que ir perfectamente entre paréntesis, por lo que es fácil ver dónde están y comprobar que no te los estás perdiendo.
Echémosles un vistazo a todos de nuevo.
Primerofor...of:
for (const elemento of arreglo) { // código a ejecutar}for:
for (inicializador; condición; expresión-final) { // código a ejecutar}while:
inicializadorwhile(condición) { // código a ejecutar expresión-final}y finalmentedo...while:
inicializadordo { // código a ejecutar expresión-final} while (condición)Nota:También hay otros tipos/características de bucle, que son útiles en situaciones avanzadas/especializadas y más allá del alcance de este artículo. Si quieres ir más allá con tu aprendizaje en bucle, lee nuestraGuía de bucles e iteraciones avanzada.
Pon a prueba tus habilidades
Has llegado al final de este artículo, pero ¿puedes recordar la información más importante? Puedes encontrar algunas pruebas adicionales para verificar que has conservado esta información antes de continuar. ConsultaPon a prueba tus habilidades: bucles.
Conclusión
Este artículo te ha revelado los conceptos básicos que hay detrás y las diferentes opciones disponibles al hacer bucles de código en JavaScript.¡Ahora deberías tener claro por qué los bucles son un buen mecanismo para lidiar con el código repetitivo y tener ganas de usarlos en tus propios ejemplos!
Si hay algo que no entendiste, vuelve a leer el artículo oponte en contacto con nosotros para pedir ayuda.
Vease también
- Bucles e iteración en detalle
- for...of referencia
- Referencia de la declaración for
- Referencias dewhile ydo...while
- Referencias debreak ycontinue