Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/midudev/preguntas-entrevista-react

Preguntas típicas sobre React para entrevistas de trabajo ⚛️
https://github.com/midudev/preguntas-entrevista-react

hacktoberfest javascript react react-entrevista-tecnica react-native react-prueba-tecnica reactjs

Last synced: 7 days ago
JSON representation

Preguntas típicas sobre React para entrevistas de trabajo ⚛️

Awesome Lists containing this project

README

        



Preguntas de entrevista para React

De cero a experto. Con respuestas detalladas en Español 🇪🇸

Deja tu :star: si te gusta el proyecto.

Streams de programación en Twitch: [twitch.tv/midudev](https://twitch.tv/midudev)

Comunidad de desarrollo en Discord: [discord.gg/midudev](https://discord.gg/midudev)

---

## Índice

- [Índice](#índice)
- [Principiante](#principiante)
- [¿Qué es React?](#qué-es-react)
- [¿Cuáles son las características principales de React?](#cuáles-son-las-características-principales-de-react)
- [¿Qué significa exactamente que sea declarativo?](#qué-significa-exactamente-que-sea-declarativo)
- [¿Qué es un componente?](#qué-es-un-componente)
- [¿Qué es JSX?](#qué-es-jsx)
- [¿Cómo se transforma el JSX?](#cómo-se-transforma-el-jsx)
- [¿Cuál es la diferencia entre componente y elemento en React?](#cuál-es-la-diferencia-entre-componente-y-elemento-en-react)
- [¿Cómo crear un componente en React?](#cómo-crear-un-componente-en-react)
- [¿Qué son las props en React?](#qué-son-las-props-en-react)
- [¿Qué es y para qué sirve la prop `children` en React?](#qué-es-y-para-qué-sirve-la-prop-children-en-react)
- [ ¿Qué diferencia hay entre props y state?](#qué-diferencia-hay-entre-props-y-state)
- [¿Se puede inicializar un estado con el valor de una prop? ¿Qué pasa si lo haces y qué hay que tener en cuenta?](#se-puede-inicializar-un-estado-con-el-valor-de-una-prop-qué-pasa-si-lo-haces-y-qué-hay-que-tener-en-cuenta)
- [¿Qué es el renderizado condicional en React?](#qué-es-el-renderizado-condicional-en-react)
- [¿Cómo puedes aplicar clases CSS a un componente en React y por qué no se puede usar `class`?](#cómo-puedes-aplicar-clases-css-a-un-componente-en-react-y-por-qué-no-se-puede-usar-class)
- [¿Cómo puedes aplicar estilos en línea a un componente en React?](#cómo-puedes-aplicar-estilos-en-línea-a-un-componente-en-react)
- [¿Cómo puedo aplicar estilos de forma condicional a un componente en React?](#cómo-puedo-aplicar-estilos-de-forma-condicional-a-un-componente-en-react)
- [¿Qué es el renderizado de listas en React?](#qué-es-el-renderizado-de-listas-en-react)
- [¿Cómo puedes escribir comentarios en React?](#cómo-puedes-escribir-comentarios-en-react)
- [¿Cómo añadir un evento a un componente en React?](#cómo-añadir-un-evento-a-un-componente-en-react)
- [¿Cómo puedo pasar un parámetro a una función que maneja un evento en React?](#cómo-puedo-pasar-un-parámetro-a-una-función-que-maneja-un-evento-en-react)
- [¿Qué es el estado en React?](#qué-es-el-estado-en-react)
- [¿Qué son los hooks?](#qué-son-los-hooks)
- [¿Qué hace el hook `useState`?](#qué-hace-el-hook-usestate)
- [¿Qué significa la expresión "subir el estado"?](#qué-significa-la-expresión-subir-el-estado)
- [¿Qué hace el hook `useEffect`?](#qué-hace-el-hook-useeffect)
- [Explica casos de uso del hook `useEffect`](#explica-casos-de-uso-del-hook-useeffect)
- [Cómo suscribirse a un evento en `useEffect`](#cómo-suscribirse-a-un-evento-en-useeffect)
- [¿Qué hace el hook `useId`?](#qué-hace-el-hook-useid)
- [¿Cómo podemos ejecutar código cuando el componente se monta?](#cómo-podemos-ejecutar-código-cuando-el-componente-se-monta)
- [¿Qué son los Fragments en React?](#qué-son-los-fragments-en-react)
- [¿Por qué es recomendable usar Fragment en vez de un div?](#por-qué-es-recomendable-usar-fragment-en-vez-de-un-div)
- [¿Qué es el Compound Components Pattern?](#qué-es-el-compound-components-pattern)
- [¿Cómo puedes inicializar un proyecto de React desde cero?](#cómo-puedes-inicializar-un-proyecto-de-react-desde-cero)
- [¿Qué es React DOM?](#qué-es-react-dom)
- [¿Qué JavaScript necesito para aprender React?](#qué-javascript-necesito-para-aprender-react)
- [JavaScript que necesitas para aprender React](#javascript-que-necesitas-para-aprender-react)
- [EcmaScript Modules o ESModules](#ecmascript-modules-o-esmodules)
- [Operador condicional (ternario)](#operador-condicional-ternario)
- [Funciones flecha o Arrow Functions](#funciones-flecha-o-arrow-functions)
- [Parámetros predeterminados (default values)](#parámetros-predeterminados-default-values)
- [Template Literals](#template-literals)
- [Propiedades abreviadas](#propiedades-abreviadas)
- [La desestructuración](#la-desestructuración)
- [Métodos de Array](#métodos-de-array)
- [Sintaxis Spread](#sintaxis-spread)
- [Operador Rest](#operador-rest)
- [Encadenamiento opcional (Optional Chaining)](#encadenamiento-opcional-optional-chaining)
- [Intermedio](#intermedio)
- [¿Cómo crear un hook personalizado (*custom hook*)?](#cómo-crear-un-hook-personalizado-custom-hook)
- [¿Cuántos `useEffect` puede tener un componente?](#cuántos-useeffect-puede-tener-un-componente)
- [¿Cómo podemos ejecutar código cuando el componente se desmonta del árbol?](#cómo-podemos-ejecutar-código-cuando-el-componente-se-desmonta-del-árbol)
- [Cómo puedes cancelar una petición a una API en `useEffect` correctamente](#cómo-puedes-cancelar-una-petición-a-una-api-en-useeffect-correctamente)
- [¿Cuáles son las reglas de los hooks en React?](#cuáles-son-las-reglas-de-los-hooks-en-react)
- [¿Qué diferencia hay entre `useEffect` y `useLayoutEffect`?](#qué-diferencia-hay-entre-useeffect-y-uselayouteffect)
- [¿Qué son mejores los componentes de clase o los componentes funcionales?](#qué-son-mejores-los-componentes-de-clase-o-los-componentes-funcionales)
- [¿Cómo mantener los componentes puros y qué ventajas tiene?](#cómo-mantener-los-componentes-puros-y-qué-ventajas-tiene)
- [¿Qué es la hidratación (hydration) en React?](#qué-es-la-hidratación-hydration-en-react)
- [¿Qué es el Server Side Rendering y qué ventajas tiene?](#qué-es-el-server-side-rendering-y-qué-ventajas-tiene)
- [¿Cómo puedes crear un Server Side Rendering con React desde cero?](#cómo-puedes-crear-un-server-side-rendering-con-react-desde-cero)
- [¿Puedes poner un ejemplo de efectos colaterales en React?](#puedes-poner-un-ejemplo-de-efectos-colaterales-en-react)
- [¿Qué diferencia hay entre componentes controlados y no controlados? ¿Qué ventajas y desventajas tienen?](#qué-diferencia-hay-entre-componentes-controlados-y-no-controlados-qué-ventajas-y-desventajas-tienen)
- [¿Qué son los High Order Components (HOC)?](#qué-son-los-high-order-components-hoc)
- [¿Qué son las render props?](#qué-son-las-render-props)
- [¿Por qué no podemos usar un `if` en el renderizado de un componente?](#por-qué-no-podemos-usar-un-if-en-el-renderizado-de-un-componente)
- [¿Por qué debemos utilizar una función para actualizar el estado de React?](#por-qué-debemos-utilizar-una-función-para-actualizar-el-estado-de-react)
- [¿Qué es el ciclo de vida de un componente en React?](#qué-es-el-ciclo-de-vida-de-un-componente-en-react)
- [¿Por qué puede ser mala práctica usar el `index` como key en un listado de React?](#por-qué-puede-ser-mala-práctica-usar-el-index-como-key-en-un-listado-de-react)
- [¿Para qué sirve el hook `useMemo`?](#para-qué-sirve-el-hook-usememo)
- [¿Es buena idea usar siempre `useMemo` para optimizar nuestros componentes?](#es-buena-idea-usar-siempre-usememo-para-optimizar-nuestros-componentes)
- [¿Para qué sirve el hook `useCallback`?](#para-qué-sirve-el-hook-usecallback)
- [¿Es buena idea usar siempre `useCallback` para optimizar nuestros componentes?](#es-buena-idea-usar-siempre-usecallback-para-optimizar-nuestros-componentes)
- [¿Cuál es la diferencia entre `useCallback` y `useMemo`?](#cuál-es-la-diferencia-entre-usecallback-y-usememo)
- [¿Qué son las refs en React?](#qué-son-las-refs-en-react)
- [¿Cómo funciona el hook `useRef`?](#cómo-funciona-el-hook-useref)
- [¿Qué hace el hook `useLayoutEffect`?](#qué-hace-el-hook-uselayouteffect)
- [Orden de ejecución del `useLayoutEffect`](#orden-de-ejecución-del-uselayouteffect)
- [¿Qué son los componentes *stateless*?](#qué-son-los-componentes-stateless)
- [¿Cómo puedes prevenir el comportamiento por defecto de un evento en React?](#cómo-puedes-prevenir-el-comportamiento-por-defecto-de-un-evento-en-react)
- [¿Qué es el `StrictMode` en React?](#qué-es-el-strictmode-en-react)
- [¿Por qué es recomendable exportar los componentes de React de forma nombrada?](#por-qué-es-recomendable-exportar-los-componentes-de-react-de-forma-nombrada)
- [¿Cómo puedes exportar múltiples componentes de un mismo archivo?](#cómo-puedes-exportar-múltiples-componentes-de-un-mismo-archivo)
- [¿Cómo puedo importar de forma dinámica un componente en React?](#cómo-puedo-importar-de-forma-dinámica-un-componente-en-react)
- [¿Cuando y por qué es recomendable importar componentes de forma dinámica?](#cuando-y-por-qué-es-recomendable-importar-componentes-de-forma-dinámica)
- [¿Sólo se pueden cargar componentes de forma dinámica si se exportan por defecto?](#sólo-se-pueden-cargar-componentes-de-forma-dinámica-si-se-exportan-por-defecto)
- [¿Qué es el contexto en React? ¿Cómo puedo crearlo y consumirlo?](#qué-es-el-contexto-en-react-cómo-puedo-crearlo-y-consumirlo)
- [¿Qué es el `SyntheticEvent` en React?](#qué-es-el-syntheticevent-en-react)
- [¿Qué es `flushSync` en React?](#qué-es-flushsync-en-react)
- [¿Qué son los Error Boundaries en React?](#qué-son-los-error-boundaries-en-react)
- [¿Qué son las Forward Refs?](#qué-son-las-forward-refs)
- [¿Cómo puedo validar el tipo de mis props?](#cómo-puedo-validar-el-tipo-de-mis-props)
- [¿Cómo puedo validar las propiedades de un objeto con PropTypes?](#cómo-puedo-validar-las-propiedades-de-un-objeto-con-proptypes)
- [¿Cómo puedo validar las propiedades de un array con PropTypes?](#cómo-puedo-validar-las-propiedades-de-un-array-con-proptypes)
- [¿Cómo puedo inyectar HTML directamente en un componente de React?](#cómo-puedo-inyectar-html-directamente-en-un-componente-de-react)
- [¿Por qué puede ser mala idea pasar siempre todas las props de un objeto a un componente?](#por-qué-puede-ser-mala-idea-pasar-siempre-todas-las-props-de-un-objeto-a-un-componente)
- [¿Cuál es el propósito del atributo "key" en React y por qué es importante usarlo correctamente al renderizar listas de elementos?](#cuál-es-el-propósito-del-atributo-key-en-react-y-por-qué-es-importante-usarlo-correctamente-al-renderizar-listas-de-elementos)
- [Experto](#experto)
- [¿Es React una biblioteca o un framework? ¿Por qué?](#es-react-una-biblioteca-o-un-framework-por-qué)
- [¿Para qué sirve el hook `useImperativeHandle`?](#para-qué-sirve-el-hook-useimperativehandle)
- [¿Para qué sirve el método `cloneElement` de React?](#para-qué-sirve-el-método-cloneelement-de-react)
- [¿Qué son los portales en React?](#qué-son-los-portales-en-react)
- [¿Por qué `StrictMode` renderiza dos veces la aplicación?](#por-qué-strictmode-renderiza-dos-veces-la-aplicación)
- [¿Qué problemas crees que pueden aparecer en una aplicación al querer visualizar listas de miles/millones de datos?](#qué-problemas-crees-que-pueden-aparecer-en-una-aplicación-al-querer-visualizar-listas-de-milesmillones-de-datos)
- [¿Cómo puedes abortar una petición fetch con `useEffect` en React?](#cómo-puedes-abortar-una-petición-fetch-con-useeffect-en-react)
- [¿Qué solución/es implementarías para evitar problemas de rendimiento al trabajar con listas de miles/millones de datos?](#qué-soluciónes-implementarías-para-evitar-problemas-de-rendimiento-al-trabajar-con-listas-de-milesmillones-de-datos)
- [Pagination](#pagination)
- [Virtualization](#virtualization)
- [¿Qué es el hook `useDebugValue`?](#qué-es-el-hook-usedebugvalue)
- [¿Qué es el `Profiler` en React?](#qué-es-el-profiler-en-react)
- [¿Cómo puedes acceder al evento nativo del navegador en React?](#cómo-puedes-acceder-al-evento-nativo-del-navegador-en-react)
- [¿Cómo puedes registrar un evento en la fase de captura en React?](#cómo-puedes-registrar-un-evento-en-la-fase-de-captura-en-react)
- [¿Cómo puedes mejorar el rendimiento del Server Side Rendering en React para evitar que bloquee el hilo principal?](#cómo-puedes-mejorar-el-rendimiento-del-server-side-rendering-en-react-para-evitar-que-bloquee-el-hilo-principal)
- [¿Qué diferencia hay entre `renderToStaticNodeStream()` y `renderToPipeableStream()`?](#qué-diferencia-hay-entre-rendertostaticnodestream-y-rendertopipeablestream)
- [¿Para qué sirve el hook `useDeferredValue`?](#para-qué-sirve-el-hook-usedeferredvalue)
- [¿Para qué sirve el método `renderToReadableStream()`?](#para-qué-sirve-el-método-rendertoreadablestream)
- [¿Cómo puedo hacer testing de un componente?](#cómo-puedo-hacer-testing-de-un-componente)
- [¿Cómo puedo hacer testing de un hook?](#cómo-puedo-hacer-testing-de-un-hook)
- [¿Qué es Flux?](#qué-es-flux)
- [Errores Típicos en React](#errores-típicos-en-react)
- [¿Qué quiere decir: Warning: Each child in a list should have a unique key prop?](#qué-quiere-decir-warning-each-child-in-a-list-should-have-a-unique-key-prop)
- [React Hook useXXX is called conditionally. React Hooks must be called in the exact same order in every component render](#react-hook-usexxx-is-called-conditionally-react-hooks-must-be-called-in-the-exact-same-order-in-every-component-render)
- [Can’t perform a React state update on an unmounted component](#cant-perform-a-react-state-update-on-an-unmounted-component)
- [Too many re-renders. React limits the number of renders to prevent an infinite loop](#too-many-re-renders-react-limits-the-number-of-renders-to-prevent-an-infinite-loop)
- [¿Qué diferencia existe entre Shadow DOM y Virtual DOM?](#qué-diferencia-existe-entre-shadow-dom-y-virtual-dom)
- [¿Qué es el Binding?](#qué-es-el-binding)

---

### Principiante

#### ¿Qué es React?

**React es una biblioteca de JavaScript de código abierto para construir interfaces de usuario.** Está basada en la componetización de la UI: la interfaz se divide en componentes independientes, que contienen su propio estado. Cuando el estado de un componente cambia, React vuelve a renderizar la interfaz.

Esto hace que React sea una herramienta muy útil para construir interfaces complejas, ya que permite dividir la interfaz en piezas más pequeñas y reutilizables.

Fue creada en 2011 por Jordan Walke, un ingeniero de software que trabajaba en Facebook y que quería simplificar la forma de crear interfaces de usuario complejas.

Es una biblioteca muy popular y es usada por muchas empresas como Facebook, Netflix, Airbnb, Twitter, Instagram, etc.

Enlaces de interés:

- [Curso de React.js](https://midu.link/react)
- [Documentación oficial de React en Español](https://es.reactjs.org/)
- [Introduction to React.js de Facebook (2013)](https://www.youtube.com/watch?v=XxVg_s8xAms)
- [Documentación oficial de React actualizada](https://beta.reactjs.org/) en inglés

**[⬆ Volver a índice](#índice)**

---

#### ¿Cuáles son las características principales de React?

Las características principales de React son:

- **Componentes**: React está basado en la componetización de la UI. La interfaz se divide en componentes independientes, que contienen su propio estado. Cuando el estado de un componente cambia, React vuelve a renderizar la interfaz.

- **Virtual DOM**: React usa un DOM virtual para renderizar los componentes. El DOM virtual es una representación en memoria del DOM real. Cuando el estado de un componente cambia, React vuelve a renderizar la interfaz. En lugar de modificar el DOM real, React modifica el DOM virtual y, a continuación, compara el DOM virtual con el DOM real. De esta forma, React sabe qué cambios se deben aplicar al DOM real.

- **Declarativo**: React es declarativo, lo que significa que no se especifica cómo se debe realizar una tarea, sino qué se debe realizar. Esto hace que el código sea más fácil de entender y de mantener.

- **Unidireccional**: React es unidireccional, lo que significa que los datos fluyen en una sola dirección. Los datos fluyen de los componentes padres a los componentes hijos.

- **Universal**: React se puede ejecutar tanto en el cliente como en el servidor. Además, puedes usar React Native para crear aplicaciones nativas para Android e iOS.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué significa exactamente que sea declarativo?

No le decimos cómo debe renderizar la interfaz a base de instrucciones. Le decimos qué debe renderizar y React se encarga de renderizarlo.

Un ejemplo entre declarativo e imperativo:

```js
// Declarativo
const element =

Hello, world

// Imperativo
const element = document.createElement('h1')
element.innerHTML = 'Hello, world'
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es un componente?

Un componente es una pieza de código que renderiza una parte de la interfaz. Los componentes pueden ser parametrizados, reutilizados y pueden contener su propio estado.

En React los componentes se crean usando funciones o clases.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es JSX?

React usa JSX para declarar qué debe renderizar. JSX es una extensión de JavaScript que permite escribir un código más cercano visualmente a HTML, que mejora la legibilidad del código y hace que sea más fácil de entender.

Sin JSX, deberíamos usar `React.createElement` para crear los elementos de la interfaz manualmente de esta forma:

```js
import { createElement } from 'react'

function Hello () { // un componente es una función! 👀
return React.createElement(
'h1', // elemento a renderizar
null, // atributos del elemento
'Hola Mundo 👋🌍!' // contenido del elemento
)
}
```

Esto es muy tedioso y poco legible. Por eso, React usa JSX para declarar qué debe renderizar. Por eso usamos JSX de esta forma:

```jsx
function Hello () {
return

Hola Mundo 👋🌍!


}
```

Ambos códigos son equivalentes.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo se transforma el JSX?

**El JSX se transforma en código JavaScript compatible en el navegador usando un *transpilador* o *compilador***. El más famoso es a día de hoy Babel, que utiliza una serie de plugins para ser compatible con la transformación, pero existen otros como SWC.

Puedes ver cómo se transforma el JSX en el [playground de código de Babel](https://babeljs.io/repl/#?browsers=defaults%2C%20not%20ie%2011%2C%20not%20ie_mob%2011&build=&builtIns=false&corejs=3.21&spec=false&loose=false&code_lz=GYVwdgxgLglg9mABACQKYBt10QCgJSIDeAUIogE6pQjlIA8AFgIwB8yc6AhogLLgAm2QLwbgaR3APBuBYfYCEdAPTMWxAL5A&debug=false&forceAllTransforms=false&shippedProposals=false&circleciRepo=&evaluate=false&fileSize=false&timeTravel=false&sourceType=module&lineWrap=true&presets=env%2Creact%2Cstage-2&prettier=false&targets=&version=7.19.5&externalPlugins=&assumptions=%7B%7D).

Hay casos especiales en los que un transpilador no es necesario. Por ejemplo, **Deno tiene soporte nativo para la sintaxis JSX** y no es necesario transformar el código para hacerlo compatible.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cuál es la diferencia entre componente y elemento en React?

Un componente es una función o clase que recibe props y devuelve un elemento.
Un elemento es un objeto que representa un nodo del DOM o una instancia de un componente de React.

```js
// Elemento que representa un nodo del DOM
{
type: 'button',
props: {
className: 'button button-blue',
children: {
type: 'b',
props: {
children: 'OK!'
}
}
}
}

// Elemento que representa una instancia de un componente
{
type: Button,
props: {
color: 'blue',
children: 'OK!'
}
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo crear un componente en React?

Los componentes en React son funciones o clases que devuelven un elemento de React. Hoy en día lo más recomendado es usar funciones:

```jsx
function HelloWorld() {
return

Hello World!


}
```

Pero también puedes usar una clase para crear un componente React:

```jsx
import { Component } from 'react'

class HelloWorld extends Component {
render() {
return

Hello World!


}
}
```

Lo importante es que el nombre de la función o clase empiece con una letra mayúscula. Esto es necesario para que React pueda distinguir entre componentes y elementos HTML.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué son las props en React?

Las props son las propiedades de un componente. Son datos que se pasan de un componente a otro. Por ejemplo, si tienes un componente `Button` que muestra un botón, puedes pasarle una prop `text` para que el botón muestre ese texto:

```jsx
function Button(props) {
return {props.text}
}
```

Podríamos entender que el componente `Button` es un botón genérico, y que la prop `text` es el texto que se muestra en el botón. Así estamos creando un componente reutilizable.

Debe considerarse además que al usar cualquier expresión JavaScript dentro de JSX debe envolverlos con `{}`, en este caso el objeto `props`, de otra forma JSX lo considerará como texto plano.

Para usarlo, indicamos el nombre del componente y le pasamos las props que queremos:

```jsx

```

Las props son una forma de parametrizar nuestros componentes igual que hacemos con las funciones. Podemos pasarle cualquier tipo de dato a un componente, incluso otros componentes.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es y para qué sirve la prop `children` en React?

La prop `children` es una prop especial que se pasa a los componentes. Es un objeto que contiene los elementos que envuelve un componente.

Por ejemplo, si tenemos un componente `Card` que muestra una tarjeta con un título y un contenido, podemos usar la prop `children` para mostrar el contenido:

```jsx
function Card(props) {
return (


{props.title}


{props.children}


)
}
```

Y luego podemos usarlo de la siguiente forma:

```jsx

Contenido de la tarjeta

```

En este caso, la prop `children` contiene el elemento `

Contenido de la tarjeta

`.

Conocer y saber usar la prop `children` es muy importante para crear componentes reutilizables en React.

**[⬆ Volver a índice](#índice)**

---

####  ¿Qué diferencia hay entre props y state?

Las *props* son un objeto que **se pasan como argumentos de un componente padre a un componente hijo**. Son inmutables y no se pueden modificar desde el componente hijo.

El *state* es un valor que **se define dentro de un componente**. Su valor es inmutable (no se puede modificar directamente) pero se puede establecer un valor nuevo del estado para que React vuelva a renderizar el componente.

Así que mientras tanto *props* como *state* afectan al renderizado del componente, su gestión es diferente.

**[⬆ Volver a índice](#índice)**

---

#### ¿Se puede inicializar un estado con el valor de una prop? ¿Qué pasa si lo haces y qué hay que tener en cuenta?

Sí, se puede inicializar el estado con el valor de una prop. Pero hay que tener en cuenta que, si la prop cambia, el estado no se actualizará automáticamente. Esto es porque el estado se inicializa una vez, cuando el componente se monta por primera vez.

Por ejemplo, con componentes funcionales:

```jsx
const Counter = () => {
const [count, setCount] = useState(0)

return (



setCount(count + 1)}>Aumentar

)
}

const Count = ({ count }) => {
const [number, setNumber] = useState(count)

return

{number}


}
```

En este caso, el componente `Count` inicializa su estado con el valor de la prop `count`. Pero si cambia el valor de la prop `count`, el estado no se actualizará automáticamente. Por lo que al hacer click, siempre veremos el número 0 en pantalla.

- [Código de ejemplo](https://stackblitz.com/edit/react-ts-cdf8n9?file=App.tsx)

En este ejemplo, lo mejor sería simplemente usar la prop `count` en el componente `Count` y así siempre se volvería a renderizar.

**Es una buena práctica evitar al máximo los estados de nuestros componentes y, siempre que se pueda, simplemente calcular el valor a mostrar a partir de las props.**

En el caso que necesites inicializar un estado con una prop, es una buena práctica añadir el prefijo de `initial` a la prop para indicar que es el valor inicial del estado y que luego no lo usaremos más:

```jsx
const Input = ({ initialValue }) => {
const [value, setValue] = useState(initialValue)

return (
setValue(e.target.value)}
/>
)
}
```

Es un error muy común pensar que la prop actualizará el estado, así que tenlo en cuenta.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es el renderizado condicional en React?

El renderizado condicional es la forma de mostrar un componente u otro dependiendo de una condición.

Para hacer renderizado condicional en React usamos el operador ternario:

```jsx
function Button({ text }) {
return text
? {text}
: null
}
```

En este caso, si la prop `text` existe, se renderiza el botón. Si no existe, no se renderiza nada.

Es común encontrar implementaciones del renderizado condicional con el operador `&&`, del tipo:

```jsx
function List({ listArray }) {
return listArray?.length && listArray.map(item=>item)
}
```

Parece que tiene sentido... si el `length` es positivo (mayor a cero) pintamos el map. !Pues no! ❌ Cuidado, si tiene `length` de cero ya que se pintará en el navegador un 0.

Es preferible utilizar el operador ternario. *Kent C. Dodds* tiene un artículo interesante hablando del tema. [Use ternaries rather than && in JSX](https://kentcdodds.com/blog/use-ternaries-rather-than-and-and-in-jsx)

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo puedes aplicar clases CSS a un componente en React y por qué no se puede usar `class`?

Para aplicar clases CSS a un componente en React usamos la prop `className`:

```jsx
function Button({ text }) {
return (

{text}

)
}
```

La razón por la que se llama `className` es porque `class` es una palabra reservada en JavaScript. Por eso, en JSX, tenemos que usar `className` para aplicar clases CSS.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo puedes aplicar estilos en línea a un componente en React?

Para aplicar estilos CSS en línea a un componente en React usamos la prop `style`. La diferencia de cómo lo haríamos con HTML, es que en React los estilos se pasan como un objeto y no como una cadena de texto (esto puede verse más claro con los dobles corchetes, los primeros para indicar que es una expresión JavaScript, y los segundos para crear el objeto):

```jsx
function Button({ text }) {
return (

{text}

)
}
```

Fíjate que, además, los nombres de las propiedades CSS están en camelCase.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo puedo aplicar estilos de forma condicional a un componente en React?

Puedes aplicar estilos de forma condicional a un componente en React usando la prop `style` y un operador ternario:

```jsx
function Button({ text, primary }) {
return (

{text}

)
}
```

En el caso anterior, si la prop `primary` es `true`, el botón tendrá el color rojo. Si no, tendrá el color azul.

También puedes seguir la misma mecánica usando clases. En este caso, usamos el operador ternario para decidir si añadir o no la clase:

```jsx
function Button({ text, primary }) {
return (

{text}

)
}
```

También podemos usar bibliotecas como `classnames`:

```jsx
import classnames from 'classnames'

function Button({ text, primary }) {
return (

{text}

)
}
```

En este caso, si la prop `primary` es `true`, se añadirá la clase `primary` al botón. Si no, no se añadirá. En cambio la clase `button` siempre se añadirá.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es el renderizado de listas en React?

El renderizado de listas es la forma de iterar un array de elementos y renderizar elementos de React para cada uno de ellos.

Para hacer renderizado de listas en React usamos el método `map` de los arrays:

```jsx
function List({ items }) {
return (


    {items.map(item => (
  • {item.name}

  • ))}

)
}
```

En este caso, se renderiza una lista de elementos usando el componente `List`. El componente `List` recibe una prop `items` que es un array de objetos del tipo `[{ id: 1, name: "John Doe" }]`. El componente `List` renderiza un elemento `li` por cada elemento del array.

El elemento `li` tiene una prop `key` que es un identificador único para cada elemento. Esto es necesario para que React pueda identificar cada elemento de la lista y actualizarlo de forma eficiente. Más adelante hay una explicación más detallada sobre esto.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo puedes escribir comentarios en React?

Si vas a escribir un comentario fuera del renderizado de un componente, puedes usar la sintaxis de comentarios de JavaScript sin problemas:

```jsx
function Button({ text }) {
// Esto es un comentario
/* Esto es un comentario
de varias líneas */

return (

{text}

)
}
```

Si vas a escribir un comentario dentro del renderizado de un componente, debes envolver el comentario en llaves y usar siempre la sintaxis de comentarios de bloque:

```jsx
function Button({ text }) {
return (

{/* Esto es un comentario en el render */}
{text}

)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo añadir un evento a un componente en React?

Para añadir un evento a un componente en React usamos la sintaxis `on` y el nombre del evento nativo del navegador en *camelCase*:

```jsx
function Button({ text, onClick }) {
return (

{text}

)
}
```

En este caso, el componente `Button` recibe una prop `onClick` que es una función. Cuando el usuario hace clic en el botón, se ejecuta la función `onClick`.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo puedo pasar un parámetro a una función que maneja un evento en React?

Para pasar un parámetro a una función que maneja un evento en React podemos usar una función anónima:

```jsx
function Button({ id, text, onClick }) {
return (
onClick(id)}>
{text}

)
}
```

Cuando el usuario hace clic en el botón, se ejecuta la función `onClick` pasándole como parámetro el valor de la prop `id`.

También puedes crear una función que ejecuta la función `onClick` pasándole el valor de la prop `id`:

```jsx
function Button({ id, text, onClick }) {
const handleClick = (event) => { // handleClick recibe el evento original
onClick(id)
}

return (

{text}

)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es el estado en React?

El estado es un objeto que contiene datos que pueden cambiar en el tiempo. En React, el estado se usa para controlar los cambios en la interfaz.

Para que entiendas el concepto, piensa en el interruptor de una habitación. Estos interruptores suelen tener dos estados: encendido y apagado. Cuando accionamos el interruptor y lo ponemos en `on` entonces la luz se enciende y cuando lo ponemos en `off` la luz se apaga.

Este mismo concepto se puede aplicar a la interfaz de usuario. Por ejemplo, el botón Me Gusta de Facebook tendría el estado de `meGusta` a `true` cuando el usuario le ha dado a Me Gusta y a `false` cuando no lo ha hecho.

No solo podemos tener en el estado valores booleanos, también podemos tener objetos, arrays, números, etc.

Por ejemplo, si tienes un componente `Counter` que muestra un contador, puedes usar el estado para controlar el valor del contador.

Para crear un estado en React usamos el hook `useState`:

```jsx
import { useState } from 'react'

function Counter() {
const [count, setCount] = useState(0)

return (


Contador: {count}


setCount(count + 1)}>Aumentar

)
}
```

Al usar el hook `useState` este devolverá un `array` de dos posiciones:

0. El valor del estado.
1. La función para cambiar el estado.

Suele usarse desestructuración para facilitar la lectura y ahorrarnos algunas líneas de código. Por otro lado, al pasarle un dato como parámetro al `useState` le estamos indicando su estado inicial.

Con un componente de clase, la creación del estado sería así:

```jsx
import { Component } from 'react'

class Counter extends Component {
constructor(props) {
super(props)
this.state = { count: 0 }
}

render() {
return (


Contador: {this.state.count}


this.setState({ count: this.state.count + 1 })}>
Aumentar


)
}
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué son los hooks?

Los Hooks son una API de React que nos permite tener estado, y otras características de React, en los componentes creados con una function.

Esto, antes, no era posible y nos obligaba a crear un componente con `class` para poder acceder a todas las posibilidades de la librería.

Hooks es gancho y, precisamente, lo que hacen, es que te permiten enganchar tus componentes funcionales a todas las características que ofrece React.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué hace el hook `useState`?

El hook `useState` es utilizado para crear variables de estado, quiere decir que su valor es dinámico, que este puede cambiar en el tiempo y eso requiere una re-renderización del componente donde se utiliza

Recibe un parámetro:

- El valor inicial de nuestra variable de estado.

Devuelve un array con dos variables:

- En primer lugar tenemos la variable que contiene el valor
- La siguiente variable es una función set, requiere el nuevo valor del estado, y este modifica el valor de la variable que anteriormente mencionamos
- Cabe destacar que la función proporciona cómo parámetro el valor actual del propio estado. Ex: `setIsOpen(isOpen => !isOpen)`

En este ejemplo mostramos como el valor de `count` se inicializa en 0, y también se renderiza cada vez que el valor es modificado con la función `setCount` en el evento `onClick` del button:

```jsx
import { useState } from 'react'

function Counter() {
const [count, setCount] = useState(0)

return (
<>

Contador: {count}


setCount(count => count + 1)}>Aumentar
>
)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué significa la expresión "subir el estado"?

Cuando varios componentes necesitan compartir los mismos datos de un estado, entonces se recomienda elevar ese estado compartido hasta su ancestro común más cercano.

Dicho de otra forma. Si dos componentes hijos comparten los mismos datos de su padre, entonces mueve el estado al padre en lugar de mantener un estado local en sus hijos.

Para entenderlo, lo mejor es que lo veamos con un ejemplo. Imagina que tenemos una lista de regalos deseados y queremos poder añadir regalos y mostrar el total de regalos que hay en la lista.

```jsx
import { useState } from 'react'

export default function App () {
return (
<>

Lista de regalos




>
)
}

function GiftList () {
const [gifts, setGifts] = useState([])

const addGift = () => {
const newGift = prompt('¿Qué regalo quieres añadir?')
setGifts([...gifts, newGift])
}

return (
<>

Regalos



    {gifts.map(gift => (
  • {gift}

  • ))}

Añadir regalo
>
)
}

function TotalGifts () {
const [totalGifts, setTotalGifts] = useState(0)

return (
<>

Total de regalos


{totalGifts}


>
)
}
```

¿Qué pasa si queremos que el total de regalos se actualice cada vez que añadimos un regalo? Como podemos ver, no es posible porque el estado de `totalGifts` está en el componente `TotalGifts` y no en el componente `GiftList`. Y como no podemos acceder al estado de `GiftList` desde `TotalGifts`, no podemos actualizar el estado de `totalGifts` cuando añadimos un regalo.

Tenemos que subir el estado de `gifts` al componente padre `App` y le pasaremos el número de regalos como prop al componente `TotalGifts`.

```jsx
import { useState } from 'react'

export default function App () {
const [gifts, setGifts] = useState([])

const addGift = () => {
const newGift = prompt('¿Qué regalo quieres añadir?')
setGifts([...gifts, newGift])
}

return (
<>

Lista de regalos




>
)
}

function GiftList ({ gifts, addGift }) {
return (
<>

Regalos



    {gifts.map(gift => (
  • {gift}

  • ))}

Añadir regalo
>
)
}

function TotalGifts ({ totalGifts }) {
return (
<>

Total de regalos


{totalGifts}


>
)
}
```

Con esto, lo que hemos hecho es *elevar el estado*. Lo hemos movido desde el componente `GiftList` al componente `App`. Ahora pasamos como prop los regalos al componente `GiftList` y una forma de actualizar el estado, y también hemos pasado como prop al componente `TotalGifts` el número de regalos.

- [Código de ejemplo](https://stackblitz.com/edit/react-ts-qitkys?file=App.tsx)

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué hace el hook `useEffect`?

El hook `useEffect` se usa para ejecutar código cuando se renderiza el componente o cuando cambian las dependencias del efecto.

Recibe dos parámetros:

- La función que se ejecutará al cambiar las dependencias o al renderizar el componente.
- Un array de dependencias. Si cambia el valor de alguna dependencia, ejecutará la función.

En este ejemplo mostramos un mensaje en consola cuando carga el componente y cada vez que cambia el valor de `count`:

```jsx
import { useEffect, useState } from 'react'

function Counter() {
const [count, setCount] = useState(0)

useEffect(() => {
console.log('El contador se ha actualizado')
}, [count])

return (
<>

Contador: {count}


setCount(count + 1)}>Aumentar
>
)
}
```

**[⬆ Volver a índice](#índice)**

---

#### Explica casos de uso del hook `useEffect`

Podemos usar el hook `useEffect` de diferentes formas, tales como:

- Ejecutar código cuando se renderiza el componente, cuando cambian las dependencias del efecto o cuando se desmonta el componente.
- Por eso puede ser útil para hacer llamadas a APIs, ya que sea nada más montar el componente o cuando cambian las dependencias.
- Realizar tracking de eventos, como Google Analytics, para saber qué páginas visitan los usuarios.
- Podemos validar un formulario para que cada vez que cambie el estado, podamos actualizar la UI y mostrar dónde están los errores.
- Podemos suscribirnos a eventos del navegador, como por ejemplo el evento `resize` para saber cuando el usuario cambia el tamaño de la ventana.

**[⬆ Volver a índice](#índice)**

---

#### Cómo suscribirse a un evento en `useEffect`

Dentro de `useEffect` nos podemos suscribir a eventos del navegador, como el evento `resize` para saber cuando el usuario cambia el tamaño de la ventana. Es importante que nos desuscribamos cuando el componente se desmonte para evitar fugas de memoria. Para ello, tenemos que devolver una función dentro del `useEffect` que se ejecutará cuando el componente se desmonte.

```jsx
import { useEffect } from 'react'

function Window() {
useEffect(() => {
const handleResize = () => {
console.log('La ventana se ha redimensionado')
}

window.addEventListener('resize', handleResize)

return () => {
window.removeEventListener('resize', handleResize)
}
}, [])

return (

Abre la consola y redimensiona la ventana


)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué hace el hook `useId`?

`useId` es un hook para generar identificadores únicos que se pueden pasar a los atributos de las etiquetas HTML y es especialmente útil para accesibilidad.

Llama `useId` en el nivel superior del componente para generar una ID única:

```jsx
import { useId } from 'react'
function PasswordField() {
const passwordHintId = useId()
// ...
```

A continuación, pasa el ID generado a diferentes atributos:

```jsx
<>


>
```

La etiqueta `aria-describedby` te permite especificar que dos etiquetas están relacionadas entre sí, puede generar una identificación única con useId donde incluso si `PasswordField` aparece varias veces en la pantalla, las identificaciones generadas no chocarán.

El ejemplo completo sería así:

```jsx
import { useId } from 'react'

function PasswordField() {
const passwordHintId = useId()

return (
<>

Password:



El password debe ser de 18 letras y contener caracteres especiales


>
)
}

export default function App() {
return (
<>

Choose password



Confirm password



>
)
}
```

Como ves en `App` estamos usando el componente dos veces. Si pusieramos una id a mano, por ejemplo `password`, entonces la ID no sería única y quedaría duplicada. Por eso es importante que generes la ID automáticamente con `useId`.

**[⬆ Volver a índice](#índice)**

---

#### ¿Cómo podemos ejecutar código cuando el componente se monta?

Podemos ejecutar código cuando el componente se monta usando el hook `useEffect` sin pasarle ninguna dependencia. En este caso, la función que se pasa como primer parámetro se ejecutará cuando el componente se monte.

```jsx
import { useEffect } from 'react'

function Component() {
useEffect(() => {
console.log('El componente se ha montado')
}, [])

return (

Abre la consola y re-dimensiona la ventana


)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué son los Fragments en React?

Los *Fragments* son una forma de agrupar elementos sin añadir un elemento extra al DOM, ya que React no permite devolver varios elementos en un componente, solo un elemento raíz.

Para crear un Fragment en React usamos el componente `Fragment`:

```jsx
import { Fragment } from 'react'

function App() {
return (

Titulo


Párrafo



)
}
```

También podemos usar la sintaxis de abreviatura:

```jsx
function App() {
return (
<>

Titulo


Párrafo


>
)
}
```

**[⬆ Volver a índice](#índice)**

---

#### ¿Por qué es recomendable usar Fragment en vez de un div?

Las razones por las que es recomendable usar Fragment en vez de un `div` a la hora de envolver varios elementos son:

- Los `div` añaden un elemento extra al DOM, mientras que los Fragments no. Esto hace que el número de elementos HTML y la profundidad del DOM sea menor.
- Los elementos envueltos con Fragment son afectados directamente por las propiedades *flex* o *grid* de CSS de su elemento padre. Si usas un `div` es posible que tengas problemas con el alineamiento de los elementos.
- Los Fragments son más rápidos que los `div` ya que no tienen que ser renderizados.
- Los `div` aplican CSS por defecto (hace que lo que envuelve el `div` se comporte como un bloque al aplicar un `display: block`) mientras que los Fragment no aplican ningún estilo por defecto.

**[⬆ Volver a índice](#índice)**

---

#### ¿Qué es el Compound Components Pattern?

Es un patrón de diseño de componentes que se basa en crear un componente padre con un solo objetivo, proporcionarle a sus hijos las propiedades necesarias para que se rendericen sin problemas.

Permite una estructura declarativa a la hora de construir nuevos componentes, además ayuda a la lectura del código por su simplicidad y limpieza.

Un ejemplo de este diseño sería una lista que renderiza los elementos hijos:

```jsx

Cat
Dog

```

```jsx
const List = ({ children, ...props }) => (


    {children}

);

const ListItem = ({ children, ...props }) => {

return (


  • {children}

  • );
    };

    export { List, ListItem };
    ```

    Este es un ejemplo sencillo, pero los componentes pueden ser tan complejos como quieras y tanto el padre como los hijos pueden tener sus propios estados.

    Enlaces de interés:

    - [Lleva tu React al siguiente nivel con Compound Pattern by dezkareid en el blog de Platzi](https://platzi.com/blog/lleva-tu-react-al-siguiente-nivel-con-compound-pattern/?utm_source=twitter&utm_medium=organic&utm_campaign=PLA_TW_BLOG_202205_react_compound_pattern)

    - [Compound Components by Jenna Smith](https://jjenzz.com/compound-components) en inglés
    - [Compound Components Lesson by Kent C. Dodds](https://egghead.io/lessons/react-write-compound-components) en inglés

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes inicializar un proyecto de React desde cero?

    Existen diversas formas de inicializar un proyecto de React desde cero. Entre las más populares están:

    - [Vite](https://vitejs.dev/guide/#scaffolding-your-first-vite-project)

    ```bash
    npm create vite@latest my-app -- --template react
    ```

    - [Create React App](https://create-react-app.dev/docs/getting-started)

    ```bash
    npx create-react-app my-app
    ```

    > La opción más popular y recomendada hoy en día es Vite. Fuente [npm trends](https://npmtrends.com/@vitejs/plugin-react-vs-create-react-app).

    Usando un Framework, entre las más populares están:

    - [Nextjs](https://nextjs.org/docs/getting-started)

    ```bash
    npx create-next-app@latest my-app
    ```

    - [Gatsby](https://www.gatsbyjs.com/docs/quick-start/)

    ```bash
    npm init gatsby
    ```

    > La opción más popular y recomendada hoy en día es Nextjs. Fuente [npm trends](https://npmtrends.com/gatsby-vs-next)

    Cada uno de ellos es un empaquetador de aplicaciones web. Se encargan de resolver las dependencias de tu proyecto, levantar un entorno de desarrollo que se refresca automáticamente con cada cambio y de empaquetar tu aplicación para producción con todos los archivos estáticos necesarios y mucho más.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es React DOM?

    React DOM es la librería que se encarga de renderizar los componentes de React para el navegador. Hay que tener en cuenta que React es una biblioteca que se puede usar en diferentes entornos (dispositivos móviles, apps de escritorio, terminal...).

    Mientras que la biblioteca de *React*, a secas, es el motor de creación de componentes, hooks, sistema de props y estado... *React DOM* es la librería que se encarga de renderizar los componentes de React específicamente en el navegador.

    *React Native*, por ejemplo, haría lo mismo, pero para dispositivos móviles.

    **[⬆ Volver a índice](#índice)**

    #### ¿Qué JavaScript necesito para aprender React?

    ##### JavaScript que necesitas para aprender React

    **Para aprender y dominar React necesitas saber JavaScript.** A diferencia de otros frameworks y bibliotecas, como *Angular* y *Vue*, que se basan en su propio *DSL* (Domain-Specific Language), React usa una extensión de la sintaxis de JavaScript llamada *JSX*. Más adelante lo veremos en detalle pero, al final, no deja de ser azúcar sintáctico para escribir menos JavaScript.

    **En React todo es JavaScript.** Para bien y para mal. Este libro da por sentados unos conocimientos previos del lenguaje de programación pero antes de empezar vamos a hacer un pequeño repaso por algunas de las características más importantes que necesitarás conocer.

    **Si ya dominas JavaScript puedes saltarte este capítulo** y continuar con el libro, pero recuerda que siempre podrás revisar este capítulo como referencia.

    ##### EcmaScript Modules o ESModules

    Los **EcmaScript Modules** es la forma nativa que tiene JavaScript para importar y exportar variables, funciones y clases entre diferentes ficheros. Hoy en día, especialmente si trabajamos con un empaquetador de aplicaciones como Webpack, vamos a estar trabajando constantemente con esta sintaxis.

    Por un lado podemos crear módulos exportándolos por defecto:

    ```js
    // sayHi.js
    // exportamos por defecto el módulo sayHi
    export default sayHi (message) {
    console.log(message)
    }

    // index.js
    // este módulo lo podremos importar con el nombre que queramos
    import sayHi from './sayHi.js'

    // al ser el módulo exportado por defecto podríamos usar otro nombre
    import miduHi from './sayHi.js'
    ```

    También podemos hacer exportaciones nombradas de módulos, de forma que un módulo tiene un nombre asignado y para importarlo necesitamos usar exactamente el nombre usado al exportarlo:

    ```js
    // sayHi.js
    // podemos usar exportaciones nombradas para mejorar esto
    export const sayHi = (message) => console.log(message)

    // y se pueden hacer tantas exportaciones de módulos nombrados como queramos
    export const anotherHi = msg => alert(msg)

    // index.js
    // ahora para importar estos módulos en otro archivo podríamos hacerlo así
    import {sayHi, anotherHi} from './sayHi.js'
    ```

    Los *imports* que hemos visto hasta aquí se conocen como *imports estáticos*. Esto significa que ese módulo será cargado en el momento de la carga del archivo que lo importa.

    También existen los *imports dinámicos*, de forma que podamos importar módulos que se carguen en el momento de la ejecución del programa o cuando nosotros decidamos (por ejemplo, como respuesta a un click).

    ```js
    document.querySelector('button').addEventListener('click', () => {
    // los imports dinámicos devuelven una Promesa
    import('./sayHi.js').then(module => {
    // ahora podemos ejecutar el módulo que hemos cargado
    module.default('Hola')
    })
    })
    ```

    Los imports dinámicos son útiles también cuando trabajamos con empaquetadores como Webpack o Vite, ya que esto creará unos *chunks* (fragmentos) que se cargarán fuera del bundle general. ¿El objetivo? Mejorar el rendimiento de la aplicación.

    Existen más sintaxis para trabajar con módulos, pero con saber las que hemos visto ya sería suficiente para seguir el libro.

    **¿Por qué es importante?**

    Para empezar React te ofrece diferentes partes de su biblioteca a través de módulos que podrás importar. Además nuestros componentes los tendremos separados en ficheros y, cada uno de ellos, se podrá importar utilizando *ESModules*.

    Además, por temas de optimización de rendimiento, podremos importar de forma dinámica componentes y así mejorar la experiencia de nuestros usuarios al necesitar cargar menos información para poder utilizar la página.

    ##### Operador condicional (ternario)

    Las ternarias son una forma de realizar condiciones sin la necesidad de usar la sintaxis con `if`. Se podría decir que es una forma de atajo para evitar escribir tanto código.

    ```js
    if (number % 2 === 0) {
    console.log('Es par')
    } else {
    console.log('Es impar')
    }

    // usando ternaria
    number % 2 === 0 ? console.log('Es par') : console.log('Es impar')
    ```

    **¿Por qué es importante?**

    En las interfaces gráficas es muy normal que, dependiendo del estado de la aplicación o los datos que nos lleguen, vamos a querer renderizar una cosa u otra en pantalla. Para realizar esto, en lugar de utilizar `if` se usan las ternarias ya que queda mucho más legible dentro del *JSX*.

    ##### Funciones flecha o Arrow Functions

    Las *funciones flecha* o *arrow function* fueron añadidas a JavaScript en el estándar ECMAScript 6 (o ES2015). En principio parece que simplemente se trata de una sintaxis alternativa más simple a la hora de crear expresiones de funciones:

    ```js
    const nombreDeLaFuncion = function (param1, param2) {
    // instrucciones de la función
    }

    const nombreDeLaFuncion = (param1, param2) => { // con arrow function
    // instrucciones de la función
    }
    ```

    Pero además del cambio de sintaxis existen otras características de las funciones flechas que se usan constantemente en React.

    ```js
    // return implícito al escribir una sola línea
    const getName = () => 'midudev'

    // ahorro de parentésis para función de un parámetro
    const duplicateNumber = num => num * 2

    // se usan mucho como callback en funciones de arrays
    const numbers = [2, 4, 6]
    const newNumbers = numbers.map(n => n / 2)
    console.log(newNumbers) // [1, 2, 3]
    ```

    También tiene algunos cambios respecto al valor de `this` pero, aunque es aconsejable dominarlo, no es realmente necesario para poder seguir con garantías el libro.

    **¿Por qué es importante?**

    Aunque hace unos años con React se trabajaba principalmente con clases, desde la irrupción de los hooks en la versión 16.8 ya no se usan mucho. Esto hace que se usen mucho más funciones.

    Las funciones flecha, además, puedes verlas fácilmente conviviendo dentro de tus componentes. Por ejemplo, a la hora de renderizar una lista de elementos ejecutarás el método `.map` del array y, como callback, seguramente usarás una función flecha anónima.

    ##### Parámetros predeterminados (default values)

    En JavaScript puedes proporcionar valores por defecto a los parámetros de una función en caso que no se le pase ningún argumento.

    ```js
    // al parámetro b le damos un valor por defecto de 1
    function multiply(a, b = 1) {
    return a * b;
    }

    // si le pasamos un argumento con valor, se ignora el valor por defecto
    console.log(multiply(5, 2)) // 10

    // si no le pasamos un argumento, se usa el valor por defecto
    console.log(multiply(5)) // 5

    // las funciones flecha también pueden usarlos
    const sayHi = (msg = 'Hola React!') => console.log(msg)
    sayHi() // 'Hola React!'
    ```

    **¿Por qué es importante?**

    En React existen dos conceptos muy importantes: **componentes y hooks**. No vamos a entrar en detalle ahora en ellos pero lo importante es que ambos son construidos con funciones.

    Poder añadir valores por defecto a los parámetros de esas funciones en el caso que no venga ningún argumento **es clave** para poder controlar React con éxito.

    Los componentes, por ejemplo, pueden no recibir parámetros y, pese a ello, seguramente vas a querer que tengan algún comportamiento por defecto. Lo podrás conseguir de esta forma.

    ##### Template Literals

    Los template literals o plantillas de cadenas llevan las cadenas de texto al siguiente nivel permitiendo expresiones incrustadas en ellas.

    ```js
    const inicio = 'Hola'
    const final = 'React'

    // usando una concatenación normal sería
    const mensaje = inicio + " " + final

    // con los template literals podemos evaluar expresiones
    const mensaje = `${inicio} ${final}`
    ```

    Como ves, para poder usar los template literals, necesitas usar el símbolo ```

    Además, nos permiten utilizar cadenas de texto de más de una línea.

    **¿Por qué es importante?**

    En React esto se puede utilizar para diferentes cosas. No sólo es normal crear cadenas de texto para mostrar en la interfaz... también puede ser útil para crear clases para tus elementos HTML de forma dinámica. Verás que los template literales están en todas partes.

    ##### Propiedades abreviadas

    Desde *ECMAScript 2015* se puede iniciar un objeto utilizado nombre de propiedades abreviadas. Esto es que si quieres utilizar como valor una variable que tiene el mismo nombre que la key, entonces puedes indicar la inicialización una vez:

    ```js
    const name = 'Miguel'
    const age = 36
    const book = 'React'

    // antes haríamos esto
    const persona = { name: name, age: age, book: book }

    // ahora podemos hacer esto, sin repetir
    const persona = { name, age, book }
    ```

    **¿Por qué es importante?**

    En React se trata muchas veces con objetos y siempre vamos a querer escribir el menor número de líneas posible para mantener nuestro código fácil de mantener y entender.

    ##### La desestructuración

    La sintaxis de *desestructuración* es una expresión de JavaScript que permite extraer valores de Arrays o propiedades de objetos en distintas variables.

    ```js
    // antes
    const array = [1, 2, 3]
    const primerNumero = array[0]
    const segundoNumero = array[1]

    // ahora
    const [primerNumero, segundoNumero] = array

    // antes con objetos
    const persona = { name: 'Miguel', age: 36, book: 'React' }
    const name = persona.name
    const age = persona.age

    // ahora con objetos
    const {age, name} = persona

    // también podemos añadir valores por defecto
    const {books = 2} = persona
    console.log(persona.books) // -> 2

    // también funciona en funciones
    const getName = ({name}) => `El nombre es ${name}`
    getName(persona)
    ```

    **¿Por qué es importante?**

    En React hay mucho código básico que da por sentado que conoces y dominas esta sintaxis. Piensa que los objetos y los arreglos son tipos de datos que son perfectos para guardar datos a representar en una interfaz. Así que poder tratarlos fácilmente te va a hacer la vida mucho más fácil.

    ##### Métodos de Array

    Saber manipular arreglos en JavaScript es básico para considerar que se domina. Cada método realiza una operación en concreto y devuelve diferentes tipos de datos. Todos los métodos que veremos reciben un callback (función) que se ejecutará para cada uno de los elementos del array.

    Vamos a revisar algunos de los métodos más usados:

    ```js
    // tenemos este array con diferentes elementos
    const networks = [
    {
    id: 'youtube',
    url: 'https://midu.tube',
    needsUpdate: true
    },
    {
    id: 'twitter',
    url: 'https://twitter.com/midudev',
    needsUpdate: true
    },
    {
    id: 'instagram',
    url: 'https://instagram.com/midu.dev',
    needsUpdate: false
    }
    ]

    // con .map podemos transformar cada elemento
    // y devolver un nuevo array
    networks.map(singleNetwork => singleNetwork.url)
    // Resultado:
    [
    'https://midu.tube',
    'https://twitter.com/midudev',
    'https://instagram.com/midu.dev'
    ]

    // con .filter podemos filtrar elementos de un array que no
    // pasen una condición determinada por la función que se le pasa.
    // Devuelve un nuevo array.
    networks.filter(singleNetwork => singleNetwork.needsUpdate === true)
    // Resultado:
    [
    { id: 'youtube', url: 'https://midu.tube', needsUpdate: true },
    { id: 'twitter', url: 'https://twitter.com/midudev', needsUpdate: true }
    ]

    // con .find podemos buscar un elemento de un array que
    // cumpla la condición definida en el callback
    networks.find(singleNetwork => singleNetwork.id === 'youtube')
    // Resultado:
    { id: 'youtube', url: 'https://midu.tube', needsUpdate: true }

    // con .some podemos revisar si algún elemento del array cumple una condición
    networks.some(singleNetwork => singleNetwork.id === 'tiktok') // false
    networks.some(singleNetwork => singleNetwork.id === 'instagram') // true
    ```

    **¿Por qué es importante?**

    En React es muy normal almacenar los datos que tenemos que representar en la UI como array. Esto hace que muchas veces necesitemos tratarlos, filtrarlos o extraer información de ellos. Es primordial entender, conocer y dominar al menos estos métodos, ya que son los más usados.

    ##### Sintaxis Spread

    La sintaxis de spread nos permite expandir un iterable o un objeto en otro lugar dónde se espere esa información. Para poder utilizarlo, necesitamos utilizar los tres puntos suspensivos `...` justo antes.

    ```js
    const networks = ['Twitter', 'Twitch', 'Instagram']
    const newNetwork = 'Tik Tok'
    // creamos un nuevo array expandiendo el array networks y
    // colocando al final el elemento newNetwork
    // utilizando la sintaxis de spread
    const allNetworks = [...networks, newNetwork]
    console.log(allNetworks)
    // -> [ 'Twitter', 'Twitch', 'Instagram', 'Tik Tok' ]
    ```

    Esto mismo lo podemos conseguir con un objeto, de forma que podemos expandir todas sus propiedades en otro objeto de forma muy sencilla.

    ```js
    const midu = { name: 'Miguel', twitter: '@midudev' }
    const miduWithNewInfo = {
    ...midu,
    youtube: 'https://youtube.com/midudev',
    books: ['Aprende React']
    }
    console.log(miduWithNewInfo)
    // {
    // name: 'Miguel',
    // twitter: '@midudev',
    // youtube: 'https://youtube.com/midudev',
    // books: [ 'Aprende React' ]
    // }
    ```

    Es importante notar que esto hace una copia, sí, pero superficial. Si tuviéramos objetos anidados dentro del objeto entonces deberíamos tener en cuenta que podríamos mutar la referencia. Veamos un ejemplo.

    ```js
    const midu = {
    name: 'Miguel',
    twitter: '@midudev',
    experience: {
    years: 18,
    focus: 'javascript'
    }
    }

    const miduWithNewInfo = {
    ...midu,
    youtube: 'https://youtube.com/midudev',
    books: ['Aprende React']
    }

    // cambiamos un par de propiedades de la "copia" del objeto
    miduWithNewInfo.name = 'Miguel Ángel'
    miduWithNewInfo.experience.years = 19

    // hacemos un console.log del objeto inicial
    console.log(midu)

    // en la consola veremos que el nombre no se ha modificado
    // en el objeto original pero los años de experiencia sí
    // ya que hemos mutado la referencia original
    // {
    // name: 'Miguel',
    // twitter: '@midudev',
    // experience: { years: 19, focus: 'javascript' }
    // }
    ```

    **¿Por qué es importante?**

    En React es muy normal tener que añadir nuevos elementos a un array o crear nuevos objetos sin necesidad de mutarlos. El operador Rest nos puede ayudar a conseguir esto. Si no conoces bien el concepto de valor y referencia en JavaScript, sería conveniente que lo repases.

    ##### Operador Rest

    La sintaxis `...` hace tiempo que funciona en JavaScript en los parámetros de una función. A esta técnica se le llamaba *parámetros rest* y nos permitía tener un número indefinido de argumentos en una función y poder acceder a ellos después como un array.

    ```js
    function suma(...allArguments) {
    return allArguments.reduce((previous, current) => {
    return previous + current
    })
    }
    ```

    Ahora el operador rest también se puede utilizar para agrupar el resto de propiedades un objeto o iterable. Esto puede ser útil para extraer un elemento en concreto del objeto o el iterable y crear una copia superficial del resto en una nueva variable.

    ```js
    const midu = {
    name: 'Miguel',
    twitter: '@midudev',
    experience: {
    years: 18,
    focus: 'javascript'
    }
    }

    const {name, ...restOfMidu} = midu

    console.log(restOfMidu)
    // -> {
    // twitter: '@midudev',
    // experience: {
    // years: 18,
    // focus: 'javascript'
    // }
    // }
    ```

    También podría funcionar con arrays:

    ```js
    const [firstNumber, ...restOfNumbers] = [1, 2, 3]
    console.log(firstNumber) // -> 1
    console.log(restOfNumbers) // -> [2, 3]
    ```

    **¿Por qué es importante?**

    Es una forma interesante de *eliminar* (de forma figurada) una propiedad de un objeto y creando una copia superficial del resto de propiedades. A veces puede ser interesante para extraer la información que queremos de unos parámetros y dejar el resto en un objeto que pasaremos hacia otro nivel.

    ##### Encadenamiento opcional (Optional Chaining)

    El operador de encadenamiento opcional `?.` te permite leer con seguridad el valor de una propiedad que está anidada dentro de diferentes niveles de un objeto.

    De esta forma, en lugar de revisar si las propiedades existen para poder acceder a ellas, lo que hacemos es usar el encadenamiento opcional.

    ```js
    const author = {
    name: 'Miguel',
    libro: {
    name: 'Aprendiendo React'
    },
    writeBook() {
    return 'Writing!'
    }
    };

    // sin optional chaining
    (author === null || author === undefined)
    ? undefined
    : (author.libro === null || author.libro === undefined)
    ? undefined
    : author.libro.name

    // con optional chaining
    author?.libro?.name
    ```

    **¿Por qué es importante?**

    Un objeto es una estructura de datos que es perfecta a la hora de representar muchos elementos de la UI. ¿Tienes un artículo? Toda la información de un artículo seguramente la tendrás representada en un objeto.

    Conforme tu UI sea más grande y compleja, estos objetos tendrán más información y necesitarás dominar el encadenamiento opcional `?.` para poder acceder a su información con garantías.

    **[⬆ Volver a índice](#índice)**

    ---

    ### Intermedio

    #### ¿Cómo crear un hook personalizado (*custom hook*)?

    Un hook personalizado es una función que empieza con la palabra `use` y que puede utilizar otros hooks. Son ideales para reutilizar lógica en diferentes componentes. Por ejemplo, podemos crear un hook personalizado para extraer la gestión del estado de un contador:

    ```js
    // ./hooks/useCounter.js

    export function useCounter() {
    const [count, setCount] = useState(0)

    const increment = () => setCount(count + 1)
    const decrement = () => setCount(count - 1)

    return { count, increment, decrement }
    }
    ```

    Para usarlo en un componente:

    ```js
    import { useCounter } from './hooks/useCounter.js'

    function Counter() {
    const { count, increment, decrement } = useCounter()

    return (
    <>
    -
    {count}
    +
    >
    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cuántos `useEffect` puede tener un componente?

    Aunque normalmente los componentes de React solo cuentan con un `useEffect` lo cierto es que podemos tener tantos `useEffect` como queramos en un componente. Cada uno de ellos se ejecutará cuando se renderice el componente o cuando cambien las dependencias del efecto.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo podemos ejecutar código cuando el componente se desmonta del árbol?

    Podemos ejecutar código cuando el componente se desmonta usando el hook `useEffect` y dentro devolver una función con el código que queremos ejecutar. En este caso, la función que se pasa como primer parámetro del `useEffect` se ejecutará cuando el componente se monte, y la función que es retornada se ejecutará cuando se desmonte.

    ```jsx
    import { useEffect } from 'react'

    function Component() {
    useEffect(() => {
    console.log('El componente se ha montado')

    return () => {
    console.log('El componente se ha desmontado')
    }
    }, [])

    return

    Ejemplo


    }
    ```

    Esto es muy útil para limpiar recursos que se hayan creado en el componente, como por ejemplo, eventos del navegador o para cancelar peticiones a APIs.

    **[⬆ Volver a índice](#índice)**

    ---

    #### Cómo puedes cancelar una petición a una API en `useEffect` correctamente

    Cuando hacemos una petición a una API, podemos cancelarla para evitar que se ejecute cuando el componente se desmonte usando `AbortController` como hacemos en este ejemplo:

    ```jsx
    useEffect(() => {
    // Creamos el controlador para abortar la petición
    const controller = new AbortController()
    // Recuperamos la señal del controlador
    const { signal } = controller
    // Hacemos la petición a la API y le pasamos como options la señal
    fetch('https://jsonplaceholder.typicode.com/posts/1', { signal })
    .then(res => res.json())
    .then(json => setMessage(json.title))
    .catch(error => {
    // Si hemos cancelado la petición, la promesa se rechaza
    // con un error de tipo AbortError
    if (error.name !== 'AbortError') {
    console.error(error.message)
    }
    })

    // Si se desmonta el componente, abortamos la petición
    return () => controller.abort()
    }, [])
    ```

    Esto también funciona con `axios`:

    ```jsx
    useEffect(() => {
    // Creamos el controlador para abortar la petición
    const controller = new AbortController()
    // Recuperamos la señal del controlador
    const { signal } = controller
    // Hacemos la petición a la API y le pasamos como options la señal
    axios
    .get('https://jsonplaceholder.typicode.com/posts/1', { signal })
    .then(res => setMessage(res.data.title))
    .catch(error => {
    // Si hemos cancelado la petición, la promesa se rechaza
    // con un error de tipo AbortError
    if (error.name !== 'AbortError') {
    console.error(error.message)
    }
    })

    // Si se desmonta el componente, abortamos la petición
    return () => controller.abort()
    }, [])
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cuáles son las reglas de los hooks en React?

    Los hooks en React tienen dos reglas fundamentales:

    - Los hooks solo se pueden usar en componentes funcionales o *custom hooks*.
    - Los hooks solo se pueden llamar en el nivel superior de un componente. No se pueden llamar dentro de bucles, condicionales o funciones anidadas.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué diferencia hay entre `useEffect` y `useLayoutEffect`?

    Aunque ambos son muy parecidos, tienen una pequeña diferencia en el momento en el que se ejecutan.

    `useLayoutEffect` se ejecuta de forma síncrona inmediatamente después que React haya actualizado completamente el DOM tras el renderizado. Puede ser útil si necesitas recuperar un elemento del DOM y acceder a sus dimensiones o posición en pantalla.

    `useEffect` se ejecuta de forma asíncrona tras el renderizado, pero no asegura que el DOM se haya actualizado. Es decir, si necesitas recuperar un elemento del DOM y acceder a sus dimensiones o posición en pantalla, no podrás hacerlo con `useEffect` porque no tienes la garantía de que el DOM se haya actualizado.

    Normalmente, el 99% de las veces, vas a querer utilizar `useEffect` y, además, tiene mejor rendimiento, ya que no bloquea el renderizado.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son mejores los componentes de clase o los componentes funcionales?

    Desde que en *React 16.8.0* se incluyeron los hooks, los componentes de funciones pueden hacer casi todo lo que los componentes de clase.

    Aunque no hay una respuesta clara a esta pregunta, normalmente los componentes funcionales son más sencillos de leer y escribir y pueden tener un mejor rendimiento en general.

    Además, **los hooks solo se pueden usar en los componentes funcionales**. Esto es importante, ya que con la creación de custom hooks podemos reutilizar la lógica y podría simplificar nuestros componentes.

    Por otro lado, los componentes de clase nos permiten usar el ciclo de vida de los componentes, algo que no podemos hacer con los componentes funcionales donde solo podemos usar `useEffect`.

    **Referencias:**

    - [Tweet de midudev donde muestra que los componentes funcionales se transpilan mejor que los de clases.](https://twitter.com/midudev/status/1065516163856310272)

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo mantener los componentes puros y qué ventajas tiene?

    Los componentes puros son aquellos que no tienen estado y que no tienen efectos secundarios. Esto quiere decir que no tienen ningún tipo de lógica que no sea la de renderizar la interfaz.

    Son más fáciles de testear y de mantener. Además, son más fáciles de entender porque no tienen lógica compleja.

    Para crear un componente puro en React usamos una function:

    ```jsx
    function Button({ text }) {
    return (

    {text}

    )
    }
    ```

    En este caso, el componente `Button` recibe una prop `text` que es un string. El componente `Button` renderiza un botón con el texto que recibe en la prop `text`.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es la hidratación (hydration) en React?

    Cuando renderizamos nuestra aplicación en el servidor, React genera un HTML estático. Este HTML estático es simplemente un string que contiene el HTML que se va a mostrar en la página.

    Cuando el navegador recibe el HTML estático, lo renderiza en la página. Sin embargo, este HTML estático no tiene interactividad. No tiene eventos, no tiene lógica, no tiene estado, etc. Podríamos decir que *no tiene vida*.

    Para hacer que este HTML estático pueda ser interactivo, React necesita que el HTML estático se convierta en un árbol de componentes de React. Esto se llama **hidratación**.

    De esta forma, en el cliente, React reutiliza este HTML estático y se dedica a adjuntar los eventos a los elementos, ejecutar los efectos que tengamos en los componentes y conciliar el estado de los componentes.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el Server Side Rendering y qué ventajas tiene?

    El *Server Side Rendering* es una técnica que consiste en renderizar el HTML en el servidor y enviarlo al cliente. Esto nos permite que el usuario vea la interfaz de la aplicación antes de que se cargue el JavaScript.

    Esta técnica nos permite mejorar la experiencia de usuario y mejorar el SEO de nuestra aplicación.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes crear un Server Side Rendering con React desde cero?

    Para crear un Server Side Rendering con React desde cero podemos usar el paquete `react-dom/server` que nos permite renderizar componentes de React en el servidor.

    Veamos un ejemplo de cómo crear un Server Side Rendering con React desde cero con Express:

    ```jsx
    import express from 'express'
    import React from 'react'
    import { renderToString } from 'react-dom/server'

    const app = express()

    app.get('/', (req, res) => {
    const html = renderToString(

    Hola mundo

    )
    res.send(html)
    })
    ```

    Esto nos devolverá el HTML de la aplicación al acceder a la ruta `/`.

    ```html

    Hola mundo


    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Puedes poner un ejemplo de efectos colaterales en React?

    Igual que las funciones en JavaScript, los componentes de React también pueden tener *side effects* (efectos colaterales). Un efecto colateral significa que el componente manipula o lee información que no está dentro de su ámbito.

    Aquí puedes ver un ejemplo simple de un componente que tiene un efecto colateral. Un componente que lee y modifica una variable que está fuera del componente. Esto hace que sea imposible saber qué renderizará el componente cada vez que se use, ya que no sabemos el valor que tendrá `count`:

    ```jsx
    let count = 0

    function Counter() {
    count = count + 1

    return (

    Contador: {count}


    )
    }

    export default function Counters() {
    return (
    <>



    >
    )
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué diferencia hay entre componentes controlados y no controlados? ¿Qué ventajas y desventajas tienen?

    A la hora de trabajar con formularios en React, tenemos dos tipos de componentes: los componentes controlados y los componentes no controlados.

    **Componentes controlados:**
    son aquellos que tienen un estado que controla el valor del componente. Por lo tanto, el valor del componente se actualiza cuando el estado cambia.

    La ventaja de este tipo de componentes es que son más fáciles de testear porque no dependen de la interfaz. También nos permiten crear validaciones muy fácilmente. La desventaja es que son más complejos de crear y mantener. Además, pueden tener un peor rendimiento, ya que provocan un re-renderizado cada vez que cambia el valor del input.

    **Componentes no controlados:** son aquellos que no tienen un estado que controle el valor del componente. El estado del componente lo controla el navegador de forma interna. Para conocer el valor del componente, tenemos que leer el valor del DOM.

    La ventaja de este tipo de componentes es que se crean de forma muy fácil y no tienes que mantener un estado. Además, el rendimiento es mejor, ya que no tiene que re-renderizarse al cambiar el valor del input. Lo malo es que hay que tratar más con el DOM directamente y crear código imperativo.

    ```js
    // Controlado:
    const [value, setValue] = useState('')
    const handleChange = () => setValue(event.target.value)

    // No controlado:

    // Usamos `inputRef.current.value` para leer el valor del input
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son los High Order Components (HOC)?

    Los High Order Components son funciones que reciben un componente como parámetro y devuelven un componente.

    ```jsx
    function withLayout(Component) {
    return function(props) {
    return




    }
    }
    ```

    En este caso, la función `withLayout` recibe un componente como parámetro y devuelve un componente. El componente devuelto renderiza el componente que se le pasa como parámetro dentro de un layout.

    Es un patrón que nos permite reutilizar código y así podemos inyectar funcionalidad, estilos o cualquier otra cosa a un componente de forma sencilla.

    Con la llegada de los hooks, los HOCs se han vuelto menos populares, pero todavía se usan en algunos casos.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son las render props?

    Son un patrón de diseño de React que nos permite reutilizar código entre componentes e inyectar información en el renderizado de los componentes.

    ```jsx
    (

    Hello {data.target}


    )}/>
    ```

    En este caso, el componente `DataProvider` recibe una función `render` como prop. Ahí le indicamos qué es lo que debe renderizar usando la información que recibe como parámetro.

    La implementación del `DataProvider` con funciones podría ser la siguiente:

    ```jsx
    function DataProvider({ render }) {
    const data = { target: 'world' }
    return render(data)
    }
    ```

    También se puede encontrar este patrón usando la prop `children` en los componentes.

    ```jsx

    {data => (

    Hello {data.target}


    )}

    ```

    Y la implementación sería similar:

    ```jsx
    function DataProvider({ children }) {
    const data = { target: 'world' }
    return children(data)
    }
    ```

    Este patrón es usado por grandes bibliotecas como `react-router`, `formik` o `react-motion`.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué no podemos usar un `if` en el renderizado de un componente?

    En React, no podemos usar un `if` en el renderizado de un componente porque no es una expresión válida de JavaScript, es una declaración. Las expresiones son aquellas que devuelven un valor y las declaraciones no devuelven ningún valor.

    En JSX solo podemos usar expresiones, por eso usamos ternarias, que sí son expresiones.

    ```jsx
    // ❌ Esto no funciona
    function Button({ text }) {
    return (

    {if (text) { return text } else { return 'Click' }}

    )
    }
    // ✅ Esto funciona
    function Button({ text }) {
    return (

    {text ? text : 'Click'}

    )
    }
    ```

    De la misma forma, tampoco podemos usar `for`, `while` o `switch` dentro del renderizado de un componente.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué debemos utilizar una función para actualizar el estado de React?

    A la hora de actualizar el estado de React, debemos utilizar la función que nos facilita el hook `useState` para actualizar el estado.

    ```jsx
    const [count, setCount] = useState(0)

    setCount(count + 1)
    ```

    ¿Por qué es esto necesario? En primer lugar, el estado en React debe ser inmutable. Es decir, no podemos modificar el estado directamente, sino que debemos siempre crear un nuevo valor para el nuevo estado.

    Esto nos permite que la integridad de la UI respecto a los datos que renderiza siempre es correcta.

    Por otro lado, llamar a una función le permite a React saber que el estado ha cambiado y que debe re-renderizar el componente si es necesario. Además esto lo hace de forma asíncrona, por lo que podemos llamar a `setCount` tantas veces como queramos y React se encargará de actualizar el estado cuando lo considere oportuno.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el ciclo de vida de un componente en React?

    En los componentes de clase, el ciclo de vida de un componente se divide en tres fases:

    - Montaje: cuando el componente se añade al DOM.
    - Actualización: cuando el componente se actualiza.
    - Desmontaje: cuando el componente se elimina del DOM.

    Dentro de este ciclo de vida, existe un conjunto de métodos que se ejecutan en el componente.

    Estos métodos se definen en la clase y se ejecutan en el orden que se muestran a continuación:

    - constructor
    - render
    - componentDidMount
    - componentDidUpdate
    - componentWillUnmount

    En cada uno de estos métodos podemos ejecutar código que nos permita controlar el comportamiento de nuestro componente.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué puede ser mala práctica usar el `index` como key en un listado de React?

    Cuando renderizamos una lista de elementos, React necesita saber qué elementos han cambiado, han sido añadidos o eliminados.

    Para ello, React necesita una key única para cada elemento de la lista. Si no le pasamos una key, React usa el índice del elemento como key.

    ```jsx
    const List = () => {
    const [items, setItems] = useState(['Item 1', 'Item 2', 'Item 3'])

    return (


      {items.map((item, index) => (
    • {item}

    • ))}

    )
    }
    ```

    En este caso, React usa el índice del elemento como `key`. Esto puede ser un problema si la lista se reordena o se eliminan elementos del array, ya que el índice de los elementos cambia.

    En este caso, React no sabe qué elementos han cambiado y puede que se produzcan errores.

    Un ejemplo donde se ve el problema:

    ```jsx
    const List = () => {
    const [items, setItems] = useState(['Item 1', 'Item 2', 'Item 3'])

    const handleRemove = (index) => {
    const newItems = [...items]
    newItems.splice(index, 1)
    setItems(newItems)
    }

    return (


      {items.map((item, index) => (

    • {item}
      handleRemove(index)}>Eliminar

    • ))}

    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el hook `useMemo`?

    El hook `useMemo` es un hook que nos permite memorizar el resultado de una función. Esto quiere decir que si la función que le pasamos como parámetro no ha cambiado, no se ejecuta de nuevo y se devuelve el resultado que ya se había calculado.

    ```jsx
    import { useMemo } from 'react'

    function Counter({ count }) {
    const double = useMemo(() => count * 2, [count])

    return (


    Contador: {count}


    Doble: {double}



    )
    }
    ```

    En este caso, el componente `Counter` recibe una prop `count` que es un número. El componente calcula el doble de ese número y lo muestra en pantalla.

    El hook `useMemo` recibe dos parámetros: una función y un array de dependencias. La función se ejecuta cuando el componente se renderiza por primera vez y cuando alguna de las dependencias cambia, en este ejemplo la prop `count`.

    La ventaja es que si la prop `count` no cambia, se evita el cálculo del doble y se devuelve el valor que ya se había calculado previamente.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Es buena idea usar siempre `useMemo` para optimizar nuestros componentes?

    No. `useMemo` es una herramienta que nos permite optimizar nuestros componentes, pero no es una herramienta mágica que nos va a hacer que nuestros componentes sean más rápidos. A veces el cálculo de un valor es tan rápido que no merece la pena memorizarlo. Incluso, en algunos casos, puede ser más lento memorizarlo que calcularlo de nuevo.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el hook `useCallback`?

    El hook `useCallback` es un hook que nos permite memorizar una función. Esto quiere decir que si la función que le pasamos como parámetro no ha cambiado, no se ejecuta de nuevo y se devuelve la función que ya se había calculado.

    ```jsx
    import { useCallback } from 'react'

    function Counter({ count, onIncrement }) {
    const handleIncrement = useCallback(() => {
    onIncrement(count)
    }, [count, onIncrement])

    return (


    Contador: {count}


    Incrementar

    )
    }
    ```

    En este caso, el componente `Counter` recibe una prop `count` que es un número y una prop `onIncrement` que es una función que se ejecuta cuando se pulsa el botón.

    El hook `useCallback` recibe dos parámetros: una función y un array de dependencias. La función se ejecuta cuando el componente se renderiza por primera vez y cuando alguna de las dependencias cambia, en este ejemplo la prop `count` o la prop `onIncrement`.

    La ventaja es que si la prop `count` o la prop `onIncrement` no cambian, se evita la creación de una nueva función y se devuelve la función que ya se había calculado previamente.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Es buena idea usar siempre `useCallback` para optimizar nuestros componentes?

    No. `useCallback` es una herramienta que nos permite optimizar nuestros componentes, pero no es una herramienta mágica que nos va a hacer que nuestros componentes sean más rápidos. A veces la creación de una función es tan rápida que no merece la pena memorizarla. Incluso, en algunos casos, puede ser más lento memorizarla que crearla de nuevo.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cuál es la diferencia entre `useCallback` y `useMemo`?

    La diferencia entre `useCallback` y `useMemo` es que `useCallback` memoriza una función y `useMemo` memoriza el resultado de una función.

    En cualquier caso, en realidad, `useCallback` es una versión especializada de `useMemo`. De hecho se puede simular la funcionalidad de `useCallback` con `useMemo`:

    ```js
    const memoizedCallback = useMemo(() => {
    return () => {
    doSomething(a, b)
    }
    }, [a, b])
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son las refs en React?

    Las refs nos permiten crear una referencia a un elemento del DOM o a un valor que se mantendrá entre renderizados. Se pueden declarar por medio del comando `createRef` o con el hook `useRef`.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo funciona el hook `useRef`?

    En el siguiente ejemplo vamos a guardar la referencia en el DOM a un elemento `` y vamos a cambiar el foco a ese elemento cuando hacemos clic en el botón.

    ```jsx
    import { useRef } from 'react'

    function TextInputWithFocusButton() {
    const inputEl = useRef(null)

    const onButtonClick = () => {
    // `current` apunta al elemento inputEl montado
    inputEl.current.focus()
    }

    return (
    <>

    Focus the input
    >
    )
    }
    ```

    Creamos una referencia `inputEl` con `useRef` y la pasamos al elemento `` como prop `ref`. Cuando el componente se monta, la referencia `inputEl` apunta al elemento `` del DOM.

    Para acceder al elemento del DOM, usamos la propiedad `current` de la referencia.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué hace el hook `useLayoutEffect`?

    `useLayoutEffect` funciona igual que el hook `useEffect`, con la excepción de que este se dispara sincrónicamente después de leer todas las mutaciones del DOM.

    Llama `useLayoutEffect` en el nivel superior del componente.

    ```jsx
    import { useLayoutEffect } from 'react';

    useLayoutEffect(() => {
    return () => {
    }
    }, []);
    ```

    `useLayoutEffect` recibe dos argumentos:

    - Una función callback que define el efecto.
    - Una lista de dependencias.

    Aunque el `useEffect` es el hook de renderizado más usado, si se necesita que los efectos del DOM muten cambiando la apariencia entre el efecto y el renderizado, entonces es conveniente que uses el `useLayoutEffect`.

    ##### Orden de ejecución del `useLayoutEffect`

    El orden de ejecución del `useLayoutEffect`, ya que se ejecuta de forma síncrona, al momento en que React termina de ejecutar todas las mutaciones, pero antes de renderizarlo en pantalla, es el siguiente:

    - El componente se actualiza por algún cambio de estado, props o el padre se re-renderiza
    - React renderiza el componente
    - Tu efecto es ejecutado
    - La pantalla se actualiza “visualmente”

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son los componentes *stateless*?

    Los componentes *stateless* son componentes que no tienen estado. Estos componentes se crean con una `function` y no tienen acceso al estado de la aplicación. La ventaja que tienen estos componentes es que hace que sea más fácil crear componentes puros (que siempre renderizan lo mismo para unas mismas props).

    ```jsx
    // Este es un ejemplo de componente stateless
    function Button({ text }) {
    return (

    {text}

    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes prevenir el comportamiento por defecto de un evento en React?

    Para prevenir el comportamiento por defecto de un evento en React, debemos usar el método `preventDefault`:

    ```jsx
    function Form({ onSubmit }) {
    const handleSubmit = (event) => {
    event.preventDefault()
    onSubmit()
    }

    return

    Enviar

    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el `StrictMode` en React?

    El `StrictMode` es un componente que nos permite activar algunas comprobaciones de desarrollo en React. Por ejemplo, detecta componentes que se renderizan de forma innecesaria o funcionalidades obsoletas que se están usando.

    ```jsx
    import { StrictMode } from 'react'

    function App() {
    return (



    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué es recomendable exportar los componentes de React de forma nombrada?

    Los componentes de React se pueden exportar de dos formas:

    - Exportación por defecto
    - Exportación nombrada

    Para exportar un componente por defecto, usamos la palabra reservada `default`:

    ```jsx
    // button.jsx
    export default function Button() {
    return Click
    }

    // App.jsx
    import Button from './button.jsx'

    function App() {
    return
    }
    ```

    La gran desventaja que tiene la exportación por defecto es que a la hora de importarlo puedes usar el nombre que quieras. Y esto trae problemas, ya que puedes no usar siempre el mismo en el proyecto o usar un nombre que no sea correcto con lo que importas.

    ```jsx
    // button.jsx
    export default function Button() {
    return Click
    }

    // App.jsx
    import MyButton from './button.jsx'

    function App() {
    return
    }

    // Otro.jsx
    import Button from './button.jsx'

    function Otro() {
    return
    }
    ```

    Los exports nombrados nos obligan a usar el mismo nombre en todos los archivos y, por tanto, nos aseguramos de que siempre estamos usando el nombre correcto.

    ```jsx
    // button.jsx
    export function Button() {
    return Click
    }

    // App.jsx
    import { Button } from './button.jsx'

    function App() {
    return
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes exportar múltiples componentes de un mismo archivo?

    Para exportar múltiples componentes de un mismo archivo, podemos usar la exportación nombrada:

    ```jsx
    // button.jsx
    export function Button({children}) {
    return {children}
    }

    export function ButtonSecondary({children}) {
    return {children}
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedo importar de forma dinámica un componente en React?

    Para importar de forma dinámica un componente en React debemos usar la función `import()`, el método `lazy()` de React y el componente `Suspense`.

    ```jsx
    // App.jsx
    import { lazy, Suspense } from 'react'

    const Button = lazy(() => import('./button.jsx'))

    export default function App() {
    return (
    Cargando botón...}>


    )
    }

    // button.jsx
    export default function Button() {
    return Botón cargado dinámicamente
    }
    ```

    Vamos a ver en detalle cada uno de los elementos que hemos usado:

    La función `import()` es parte del estándar de ECMAScript y nos permite importar de forma dinámica un módulo. Esta función devuelve una promesa que se resuelve con el módulo importado.

    El método `lazy()` de React nos permite crear un componente que se renderiza de forma diferida. Este método recibe una función que debe devolver una promesa que se resuelve con un componente. En este caso, se resolverá con el componente que tenemos en el fichero `button.jsx`. Ten en cuenta que el componente que devuelve `lazy()` **debe ser un componente de React y ser exportado por defecto** (`export default`).

    El componente `Suspense` nos permite mostrar un mensaje mientras se está cargando el componente. Este componente recibe una prop `fallback` que es el mensaje que se muestra mientras se está cargando el componente.

    - [Código de ejemplo](https://stackblitz.com/edit/react-ts-n6zal2?file=App.tsx)

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cuando y por qué es recomendable importar componentes de forma dinámica?

    En React, nuestras aplicaciones están creadas a partir de componentes. Estos componentes se pueden importar de forma **estática o dinámica**.

    La importación de componentes de forma estática es la forma más común de importar componentes en React. En este caso, los componentes se importan en la parte superior del fichero y se renderizan en el código. El problema es que, si siempre lo hacemos así, es bastante probable que estemos cargando componentes que no se van a usar desde el principio.

    ```jsx
    import { useState } from 'react'
    // importamos de forma estática el componente de la Modal
    import { SuperBigModal } from './super-big-modal.jsx'

    // mostrar modal si el usuario da click en un botón
    export default function App () {
    const [showModal, setShowModal] = useState(false)

    return (


    setShowModal(true)}>Mostrar modal
    {showModal && }

    )
    }
    ```

    Este componente `SuperBigModal` se importa de forma estática, por lo que se carga desde el principio. Pero, ¿qué pasa si el usuario no da click en el botón para mostrar la modal? En este caso, está cargando el componente pese a que no lo está usando.

    Si queremos ofrecer la mejor experiencia a nuestros usuarios, debemos intentar que la aplicación cargue lo más rápido posible. Por eso, es recomendable importar de forma dinámica los componentes que no se van a usar desde el principio.

    ```jsx
    import { useState, lazy, Suspense } from 'react'
    // importamos de forma dinámica el componente de la Modal
    const SuperBigModal = lazy(() => import('./super-big-modal.jsx'))

    // mostrar modal si el usuario da click en un botón
    export default function App () {
    const [showModal, setShowModal] = useState(false)

    return (


    setShowModal(true)}>Mostrar modal
    Cargando modal...
    }>
    {showModal && }


    )
    }
    ```

    De esta forma, la parte de código que importa el componente `SuperBigModal` se carga de forma dinámica, es decir, cuando el usuario da click en el botón para mostrar la modal.

    Dependiendo del empaquetador de aplicaciones web que uses y su configuración, es posible que el resultado de la carga sea diferente (algunos creará un archivo a parte del *bundle* principal, otros podrían hacer un streaming del HTML...) pero la intención del código es la misma.

    Así que siempre debemos intentar cargar los componentes de forma dinámica cuando no se vayan a usar desde el principio, sobretodo cuando están detrás de la interacción de un usuario. Lo mismo podría ocurrir con rutas completas de nuestra aplicación. ¿Por qué cargar la página de *About* si el usuario está visitando la página principal?

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Sólo se pueden cargar componentes de forma dinámica si se exportan por defecto?

    No, no es necesario que los componentes se exporten por defecto para poder cargarlos de forma dinámica. Podemos exportarlos de forma nombrada y cargarlos de forma dinámica... pero no es lo más recomendable ya que el código necesario es mucho más lioso.

    ```jsx
    // button.jsx
    // exportamos el componente Button de forma nombrada
    export function Button() {
    return Botón cargado dinámicamente
    }

    // app.jsx
    import { lazy, Suspense } from 'react'

    // Al hacer el import dinámico, debemos especificar el nombre del componente que queremos importar
    // y hacer que devuelva un objeto donde la key default pasar a ser el componente nombrado
    const Button = lazy(
    () => import('./button.jsx')
    .then(({Button}) => ({ default: Button }))
    )

    export default function App () {
    return (


    Cargando botón...
    }>



    )
    }
    ```

    Otra opción es tener un fichero intermedio que exporte el componente de forma por defecto y que sea el que importemos de forma dinámica.

    ```jsx
    // button-component.jsx
    // exportamos el componente Button de forma nombrada
    export function Button() {
    return Botón cargado dinámicamente
    }

    // button.jsx
    export { Button as default } from './button-component.jsx'

    // app.jsx
    import { lazy, Suspense } from 'react'

    const Button = lazy(() => import('./button.jsx'))

    export default function App () {
    return (


    Cargando botón...
    }>



    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el contexto en React? ¿Cómo puedo crearlo y consumirlo?

    El contexto es una forma de pasar datos a través de la jerarquía de componentes sin tener que pasar props manualmente en cada nivel.

    Para crear un contexto en React usamos el hook `createContext`:

    ```jsx
    import { createContext } from 'react'

    const ThemeContext = createContext()
    ```

    Para usar el contexto, debemos envolver el árbol de componentes con el componente `Provider`:

    ```jsx

    ```

    Para consumir el contexto, debemos usar el hook `useContext`:

    ```jsx
    import { useContext } from 'react'

    function Button() {
    const theme = useContext(ThemeContext)
    return Haz clic aquí
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el `SyntheticEvent` en React?

    El `SyntheticEvent` es una abstracción del evento nativo del navegador. Esto le permite a React tener un comportamiento consistente en todos los navegadores.

    Dentro del `SyntheticEvent` puede encontrarse una referencia al evento nativo en su atributo `nativeEvent`

    ```jsx
    function App() {
    function handleClick(event) {
    console.log(event)
    }

    return Haz clic aquí
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es `flushSync` en React?

    `flushSync(callback)` Obliga a React a ejecutar de manera síncrona todas las actualizaciones de los state dentro del callback proporcionado. Así se asegura que el DOM se actualiza inmediatamente.

    ```jsx
    import { flushSync } from "react-dom"

    function App() {
    const handleClick = () => {
    setId(1)
    // component no hace re-render 🚫
    flushSync(() => {
    setId(2)
    // component re-renderiza aquí 🔄
    })
    // component ha sido re-renderizado y el DOM ha sido actualizado ✅
    flushSync(() => {
    setName("John")
    // component no hace re-render 🚫
    setEmail("[email protected]")
    // component re-renderiza aquí 🔄
    })
    // component ha sido re-renderizado y el DOM ha sido actualizado ✅
    }

    return Haz clic aquí
    }
    ```

    NOTA: `flushSync` puede afectar significativamente el rendimiento. Úsalo con moderación.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son los Error Boundaries en React?

    Los Error Boundaries son componentes que nos permiten manejar los errores que se producen en el árbol de componentes. Para crear un Error Boundary, debemos crear un componente que implemente el método `componentDidCatch`:

    ```jsx
    class ErrorBoundary extends React.Component {
    constructor(props) {
    super(props)
    this.state = { hasError: false }
    }

    static getDerivedStateFromError(error) {
    return { hasError: true }
    }

    componentDidCatch(error, errorInfo) {
    console.log(error, errorInfo)
    }

    render() {
    if (this.state.hasError) {
    return

    Algo ha ido mal


    }

    return this.props.children
    }
    }
    ```

    De esta forma podemos capturar los errores que se producen en el árbol de componentes y mostrar un mensaje de error personalizado mientras evitamos que nuestra aplicación se rompa completamente.

    Ahora podemos envolver el árbol de componentes con el componente `ErrorBoundary`:

    ```jsx

    ```

    Podemos crear un Error Boundary en cualquier nivel del árbol de componentes, de esta forma podemos tener un control más granular de los errores.

    ```jsx




    ```

    Por ahora no existe una forma nativa de crear un Error Boundary en una función de React. Para crear un Error Boundary en una función, puedes usar la librería [react-error-boundary](https://github.com/bvaughn/react-error-boundary).

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son las Forward Refs?

    El reenvío de referencia o *Forward Refs* es una técnica que nos permite acceder a una referencia de un componente hijo desde un componente padre.

    ```jsx
    // Button.jsx
    import { forwardRef } from 'react'

    export const Button = forwardRef((props, ref) => (

    {props.children}

    ));

    // Parent.jsx
    import { Button } from './Button'
    import { useRef } from 'react'

    const Parent = () => {
    const ref = useRef()

    useEffect(() => {
    // Desde el padre podemos hacer focus
    // al botón que tenemos en el hijo
    ref.current?.focus?.()
    }, [ref.current])

    return (
    My button
    )
    }
    ```

    En este ejemplo, recuperamos la referencia del botón (elemento HTML ``) y la recupera el componente padre (`Parent`), para poder hacer focus en él gracias al uso de `forwardRef` en el componente hijo (`Button`).

    Para la gran mayoría de componentes esto no es necesario pero puede ser útil para sistemas de diseño o componentes de terceros reutilizables.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedo validar el tipo de mis props?

    React proporciona una forma de validar el tipo de las props de un componente en tiempo de ejecución y en modo desarrollo. Esto es útil para asegurarnos de que los componentes se están utilizando correctamente.

    El paquete se llama `prop-types` y se puede instalar con `npm install prop-types`.

    ```jsx
    import PropTypes from "prop-types"

    function App(props) {
    return

    {props.title}


    }

    App.propTypes = {
    title: PropTypes.string.isRequired,
    }
    ```

    En este ejemplo, estamos validando que la prop `title` sea de tipo `string` y que sea obligatoria.

    Existen una colección de *PropTypes* ya definidas para ayudarte a comprobar los tipos de las props más comunes:

    ```js
    PropTypes.number // número
    PropTypes.string // string
    PropTypes.array // array
    PropTypes.object // objeto
    PropTypes.bool // un booleano
    PropTypes.func // función
    PropTypes.node // cualquier cosa renderizable en React, como un número, string, elemento, array, etc.
    PropTypes.element // un elemento React
    PropTypes.symbol // un Symbol de JavaScript
    PropTypes.any // cualquier tipo de dato
    ```

    A todas estas se le puede añadir la propiedad `isRequired` para indicar que es obligatoria.

    > Otra opción es usar TypeScript, un lenguaje de programación que compila a JavaScript y que ofrece validación de tipos de forma estática. Ten en cuenta que mientras que TypeScript comprueba los tipos en tiempo de compilación, las PropTypes lo hacen en tiempo de ejecución.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedo validar las propiedades de un objeto con PropTypes?

    Para validar las propiedades de un objeto que se pasa como prop, podemos usar la propiedad `shape` de `PropTypes`:

    ```jsx
    import PropTypes from "prop-types"

    function App({ title }) {
    const { text, color } = title
    return

    {text}


    }

    App.propTypes = {
    title: PropTypes.shape({
    text: PropTypes.string.isRequired,
    color: PropTypes.string.isRequired,
    }),
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedo validar las propiedades de un array con PropTypes?

    Para validar las propiedades de un array que se pasa como prop, podemos usar la propiedad `arrayOf` de `PropTypes`:

    ```jsx
    import PropTypes from "prop-types"

    function App({ items }) {
    return (


      {items.map((item) => (
    • {item.text}

    • ))}

    )
    }

    App.propTypes = {
    items: PropTypes.arrayOf(
    PropTypes.shape({
    text: PropTypes.string.isRequired,
    })
    ).isRequired,
    }
    ```

    En este caso estamos validando que `items` sea un array y que cada uno de sus elementos sea un objeto con la propiedad `text` de tipo `string`. Además, la prop es obligatoria.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedo inyectar HTML directamente en un componente de React?

    Una de las razones por las que se creó React es para evitar los ataques XSS (*Cross-Site Scripting*), impidiendo que un usuario pueda inyectar código HTML en la página.

    Por ello, React al intentar evaluar un string que contiene HTML lo escapa automáticamente. Por ejemplo, si intentamos renderizar el siguiente string:

    ```jsx
    const html = "

    My title

    "

    function App() {
    return

    {html}

    }
    ```

    Veremos que en lugar de renderizar el HTML, lo escapa:

    ```html

    <h1>My title</h1>

    ```

    Sin embargo, hay ocasiones en las que es necesario inyectar HTML directamente en un componente. Ya sea por traducciones que tenemos, porque viene el HTML desde el servidor y ya viene saneado, o por un componente de terceros.

    Para ello, podemos usar la propiedad `dangerouslySetInnerHTML`:

    ```jsx
    const html = "

    My title

    "

    function App() {
    return


    }
    ```

    Ahora sí veremos el HTML renderizado:

    ```html

    My title


    ```

    Como ves, **el nombre ya nos indica que es una propiedad peligrosa y que debemos usarla con cuidado.** Intenta evitarla siempre que puedas y sólo recurre a ella cuando realmente no tengas otra opción.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué puede ser mala idea pasar siempre todas las props de un objeto a un componente?

    Digamos que tenemos un componente `App` que recibe un objeto `props` con todas las props que necesita:

    ```jsx
    function App(props) {
    return

    {props.title}


    }
    ```

    Y que tenemos otro componente `Layout` que recibe un objeto `props` con todas las props que necesita:

    ```jsx
    function Layout(props) {
    return (




    )
    }
    ```

    En este caso, `Layout` está pasando todas las props que recibe a `App`. Esto puede ser una mala idea por varias razones:

    - Si `Layout` recibe una prop que no necesita, la pasará a `App` y éste puede que no la use. Esto puede ser confuso para el que lea el código.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cuál es el propósito del atributo "key" en React y por qué es importante usarlo correctamente al renderizar listas de elementos?

    El propósito del atributo "key" en React es proporcionar una identificación única a cada elemento en una lista renderizada dinámicamente. Esto permite a React identificar qué elementos han cambiado, añadido o eliminado de la lista cuando se realiza una actualización.

    Cuando se renderiza una lista en React sin el atributo "key", React puede tener dificultades para identificar correctamente los cambios en la lista, lo que puede resultar en un comportamiento inesperado, como la re-renderización innecesaria de elementos o la pérdida de estado de los componentes.

    Por lo tanto, es importante utilizar el atributo "key" de manera correcta y única para cada elemento de la lista, preferiblemente utilizando identificadores únicos de cada elemento en lugar de índices de array, para garantizar un rendimiento óptimo y un comportamiento predecible en la aplicación.

    Ejemplo de cómo utilizar el atributo "key" en React:

    ```jsx
    import React from 'react';

    const ListaItems = ({ items }) => {
    return (


      {items.map((item) => (
    • {item.nombre}

    • ))}

    );
    };

    export default ListaItems;
    ```

    ---

    **[⬆ Volver a índice](#índice)**

    ### Experto

    #### ¿Es React una biblioteca o un framework? ¿Por qué?

    Existe una fina línea hoy en día entre qué es una biblioteca o un framework. Oficialmente, React se autodenomina como biblioteca. Esto es porque para poder crear una aplicación completa, necesitas usar otras bibliotecas.

    Por ejemplo, *React* no ofrece un sistema de enrutado de aplicaciones oficial. Por ello, hay que usar una biblioteca como [React Router](https://reactrouter.com/) o usar un *framework* como [Next.js](https://nextjs.org/) que ya incluye un sistema de enrutado.

    Tampoco puedes usar React para añadir las cabeceras que van en el `` en tu aplicación, y también necesitarás otra biblioteca o framework para solucionar esto.

    Otra diferencia es que React no está opinionado sobre qué empaquetador de aplicaciones usar. En cambio `Angular` en su propio tutorial ya te indica que debes usar `@angular/cli` para crear una aplicación, en cambio React siempre te deja la libertad de elegir qué empaquetador usar y ofrece diferentes opciones.

    Aún así, existe gente que considera a React como un framework. Aunque no hay una definición oficial de qué es un framework, la mayoría de la gente considera que un framework es una biblioteca que incluye otras bibliotecas para crear una aplicación completa de forma opinionada y casi sin configuración.

    Por ejemplo, **Next.js se podría considerar un framework de React** porque incluye React, un sistema de enrutado, un sistema de renderizado del lado del servidor, etc.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el hook `useImperativeHandle`?

    Nos permite definir qué propiedades y métodos queremos que sean accesibles desde el componente padre.

    En el siguiente ejemplo vamos a crear un componente `TextInput` que tiene un método `focus` que cambia el foco al elemento ``.

    ```jsx
    import { useRef, useImperativeHandle } from 'react'

    function TextInput(props, ref) {
    const inputEl = useRef(null)

    useImperativeHandle(ref, () => ({
    focus: () => {
    inputEl.current.focus()
    }
    }))

    return (

    )
    }
    ```

    Creamos una referencia `inputEl` con `useRef` y la pasamos al elemento `` como prop `ref`. Cuando el componente se monta, la referencia `inputEl` apunta al elemento `` del DOM.

    Para acceder al elemento del DOM, usamos la propiedad `current` de la referencia.

    Para que el componente padre pueda acceder al método `focus`, usamos el hook `useImperativeHandle`. Este hook recibe dos parámetros: una referencia y una función que devuelve un objeto con las propiedades y métodos que queremos que sean accesibles desde el componente padre.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el método `cloneElement` de React?

    Te permite clonar un elemento React y añadirle o modificar las props que recibe.

    ```jsx
    import { cloneElement } from 'react'

    const Hello = ({ name }) =>

    Hello {name}

    const App = () => {
    const element =

    return (


    {cloneElement(element, { name: 'TMChein' })}
    {cloneElement(element, { name: 'Madeval' })}
    {cloneElement(element, { name: 'Gorusuke' })}

    )
    }
    ```

    En este ejemplo, clonamos `element` que tenía la prop `midudev` y le pasamos una prop `name` diferente cada vez. Esto renderizará tres veces el componente `Hello` con los nombres `TMChein`, `Madeval` y `Gorusuke`. Sin rastro de la prop original.

    Puede ser útil para modificar un elemento que ya nos viene de un componente padre y del que no tenemos posibilidad de re-crear con el componente.

    - [Código de ejemplo](https://stackblitz.com/edit/react-ts-tc39vr?file=App.tsx)

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué son los portales en React?

    Los portales nos permiten renderizar un componente en un nodo del DOM que no es hijo del componente que lo renderiza.

    Es perfecto para ciertos casos de uso como, por ejemplo, modales:

    ```jsx
    import { createPortal } from 'react-dom'

    function Modal() {
    return createPortal(


    Modal


    ,
    document.getElementById('modal')
    )
    }
    ```

    `createPortal` acepta dos parámetros:

    - El primer parámetro es el componente que queremos renderizar
    - El segundo parámetro es el nodo del DOM donde queremos renderizar el componente

    En este caso el modal se renderiza en el nodo `#modal` del DOM.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Por qué `StrictMode` renderiza dos veces la aplicación?

    Cuando el modo `StrictMode` está activado, React monta los componentes dos veces (el estado y el DOM se preserva). Esto ayuda a encontrar efectos que necesitan una limpieza o expone problemas con *race conditions*.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué problemas crees que pueden aparecer en una aplicación al querer visualizar listas de miles/millones de datos?

    - **Tiempo de respuesta del servidor:** Hacer peticiones de millones de datos no es, en general, una buena estrategia. Incluso en el mejor de los casos, en el que el servidor solo debe devolver los datos sin tratarlos, hay un coste asociado al *parseo* y *envío* de los mismos a través de la red. Llamadas con un tamaño desmesurado pueden incurrir en interfaces lentas, e incluso en *timeouts* en la respuesta.
    - **Problemas de rendimiento:** Aunque es cierto que **React** se basa en un modelo *declarativo* en el cual no debemos tener una exhaustivo control o gestión de cómo se *renderiza*, no hay que olvidar que malas decisiones técnicas pueden conllevar aplicaciones totalmente inestables incluso con las mejores tecnologías. No es viable *renderizar* un *DOM* con millones de elementos, el *navegador* no podrá gestionarlo y, tarde o temprano, la aplicación no será usable.

    Como developers, nuestra misión es encontrar el equilibrio entre rendimiento y experiencia, intentando priorizar siempre cómo el usuario sentirá la aplicación. No hay ningún caso lo suficientemente justificado para *renderizar* en pantalla miles de datos.

    **El espacio de visualización es limitado (*viewport*), al igual que deberían serlo los datos que añadimos al DOM.**

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes abortar una petición fetch con `useEffect` en React?

    Si quieres evitar que exista una *race condition* entre una petición asíncrona y que el componente se desmonte, puedes usar la API de `AbortController` para abortar la petición cuando lo necesites:

    ```jsx
    import { useEffect, useState } from 'react'

    function Movies () {
    const [movies, setMovies] = useState([])

    useEffect(() => {
    // creamos un controlador para abortar la petición
    const abortController = new AbortController()

    // pasamos el signal al fetch para que sepa que debe abortar
    fetchMovies({ signal: abortController.signal })
    .then(() => {
    setMovies(data.results)
    }).catch(error => {
    if (error.name === 'AbortError') {
    console.log('fetch aborted')
    }
    })

    return () => {
    // al desmontar el componente, abortamos la petición
    // sólo funcionará si la petición sigue en curso
    abortController.abort()
    }
    })

    // ...
    }

    // Debemos pasarle el parámetro signal al `fetch`
    // para que enlace la petición con el controlador
    const fetchMovies = ({ signal }) => {
    return fetch('https://api.themoviedb.org/3/movie/popular', {
    signal // <--- pasamos el signal
    }).then(response => response.json())
    }
    ```

    De esta forma evitamos que se produzca un error por parte de React de intentar actualizar el estado de un componente que ya no existe, además de evitar que se produzcan llamadas innecesarias al servidor.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué solución/es implementarías para evitar problemas de rendimiento al trabajar con listas de miles/millones de datos?

    ##### Pagination

    En lugar de recibir la lista en una sola llamada a la API (lo cual sería negativo tanto para el rendimiento como para el propio servidor y tiempo de respuesta de la API), podríamos implementar un sistema de paginación en el cual la API recibirá un *offset* o *rango* de datos deseados. En el FE nuestra responsabilidad es mostrar unos controles adecuados (interfaz de paginación) y gestionar las llamadas a petición de cambio de página para siempre limitar la cantidad de DOM renderizado evitando así una sobrecarga del *DOM* y, por lo tanto, problemas de rendimiento.

    ##### Virtualization

    Existe una técnica llamada *Virtualización* que gestiona cuántos elementos de una lista mantenemos ***vivos*** en el *DOM*. El concepto se basa en solo montar los elementos que estén dentro del *viewport* más un *buffer* determinado (para evitar falta de datos al hacer scroll) y, en cambio, desmontar del *DOM* todos aquellos elementos que estén fuera de la vista del usuario. De este modo podremos obtener lo mejor de los dos mundos, una experiencia integrada y un DOM liviano que evitará posibles errores de rendimiento. Con esta solución también podremos aprovechar que contamos con los datos en memoria para realizar búsquedas/filtrados sin necesidad de más llamadas al servidor.

    Puedes consultar esta librería para aplicar Virtualización con React: [React Virtualized](https://github.com/bvaughn/react-virtualized).

    Hay que tener en cuenta que cada caso de uso puede encontrar beneficios y/o perjuicios en ambos métodos, dependiendo de factores como capacidad de respuesta de la API, cantidad de datos, necesidad de filtros complejos, etc. Por ello es importante analizar cada caso con criterio.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el hook `useDebugValue`?

    Nos permite mostrar un valor personalizado en la pestaña de *React DevTools* que nos permitirá depurar nuestro código.

    ```jsx
    import { useDebugValue } from 'react'

    function useCustomHook() {
    const value = 'custom value'
    useDebugValue(value)
    return value
    }
    ```

    En este ejemplo, el valor personalizado que se muestra en la pestaña de *React DevTools* es `custom value`.

    Aunque es útil para depurar, no se recomienda usar este hook en producción.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el `Profiler` en React?

    El `Profiler` es un componente que nos permite medir el tiempo que tarda en renderizarse un componente y sus hijos.

    ```jsx
    import { Profiler } from 'react'

    function App() {
    return (
    {
    console.log({id, phase, actualDuration})
    }}>


    )
    }
    ```

    El componente `Profiler` recibe dos parámetros:

    - `id`: es un identificador único para el componente
    - `onRender`: es una función que se ejecuta cada vez que el componente se renderiza

    Esta información es muy útil para detectar componentes que toman mucho tiempo en renderizarse y optimizarlos.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes acceder al evento nativo del navegador en React?

    React no expone el evento nativo del navegador. En su lugar, React crea un objeto sintético que se basa en el evento nativo del navegador llamado `SyntheticEvent`. Para acceder al evento nativo del navegador, debemos usar el atributo `nativeEvent`:

    ```jsx
    function Button({ onClick }) {
    return onClick(e.nativeEvent)}>Haz clic aquí
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes registrar un evento en la fase de captura en React?

    En React, los eventos se registran en la fase de burbuja por defecto. Para registrar un evento en la fase de captura, debemos añadir `Capture` al nombre del evento:

    ```jsx
    function Button({ onClick }) {
    return Haz clic aquí
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Cómo puedes mejorar el rendimiento del Server Side Rendering en React para evitar que bloquee el hilo principal?

    Aunque puedes usar el método `renderToString` para renderizar el HTML en el servidor, este método es síncrono y bloquea el hilo principal. Para evitar que bloquee el hilo principal, debemos usar el método `renderToPipeableStream`:

    ```jsx
    let didError = false
    const stream = renderToPipeableStream(
    ,
    {
    onShellReady() {
    // El contenido por encima de los límites de Suspense ya están listos
    // Si hay un error antes de empezar a hacer stream, mostramos el error adecuado
    res.statusCode = didError ? 500 : 200
    res.setHeader('Content-type', 'text/html')
    stream.pipe(res)
    },
    onShellError(error) {
    // Si algo ha ido mal al renderizar el contenido anterior a los límites de Suspense, lo indicamos.
    res.statusCode = 500
    res.send(
    '

    Loading...

    '
    )
    },
    onAllReady() {
    // Si no quieres hacer streaming de los datos, puedes usar
    // esto en lugar de onShellReady. Esto se ejecuta cuando
    // todo el HTML está listo para ser enviado.
    // Perfecto para crawlers o generación de sitios estáticos

    // res.statusCode = didError ? 500 : 200
    // res.setHeader('Content-type', 'text/html')
    // stream.pipe(res)
    },
    onError(err) {
    didError = true
    console.error(err)
    },
    }
    )
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué diferencia hay entre `renderToStaticNodeStream()` y `renderToPipeableStream()`?

    `renderToStaticNodeStream()` devuelve un stream de nodos estáticos, esto significa que no añade atributos extras para el DOM que React usa internamente para poder lograr la hidratación del HTML en el cliente. Esto significa que no podrás hacer el HTML interactivo en el cliente, pero puede ser útil para páginas totalmente estáticas.

    `renderToPipeableStream()` devuelve un stream de nodos que contienen atributos del DOM extra para que React pueda hidratar el HTML en el cliente. Esto significa que podrás hacer el HTML interactivo en el cliente pero puede ser más lento que `renderToStaticNodeStream()`.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el hook `useDeferredValue`?

    El hook `useDeferredValue` nos permite renderizar un valor con una prioridad baja. Esto es útil para renderizar un valor que no es crítico para la interacción del usuario.

    ```jsx
    function App() {
    const [text, setText] = useState('¡Hola mundo!')
    const deferredText = useDeferredValue(text, { timeoutMs: 2000 })

    return (


    {/* Seguimos pasando el texto actual como valor del input */}

    ...
    {/* Pero la lista de resultados se podría renderizar más tarde si fuera necesario */}


    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Para qué sirve el método `renderToReadableStream()`?

    Este método es similar a `renderToNodeStream`, pero está pensado para entornos que soporten Web Streams como `Deno`.

    Un ejemplo de uso sería el siguiente:

    ```jsx
    const controller = new AbortController()
    const { signal } = controller

    let didError = false

    try {
    const stream = await renderToReadableStream(

    Success
    ,
    {
    signal,
    onError(error) {
    didError = true
    console.error(error)
    }
    }
    )

    // Si quieres enviar todo el HTML en vez de hacer streaming, puedes usar esta línea
    // Es útil para crawlers o generación estática:
    // await stream.allReady

    return new Response(stream, {
    status: didError ? 500 : 200,
    headers: {'Content-Type': 'text/html'},
    })
    } catch (error) {
    return new Response(
    '

    Loading...

    ',
    {
    status: 500,
    headers: {'Content-Type': 'text/html'},
    }
    )
    }
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    ### ¿Cómo puedo hacer testing de un componente?

    Para hacer testing de un componente, puedes usar la función `render` de la librería `@testing-library/react`. Esta función nos permite renderizar un componente y obtener el resultado.

    ```jsx
    import { render } from '@testing-library/react'

    function Counter() {
    const [count, setCount] = useState(0)
    const increment = () => setCount(count + 1)
    return (


    Count: {count}


    Increment

    )
    }

    test('Counter', () => {
    const { getByText } = render()

    expect(getByText('Count: 0')).toBeInTheDocument()
    fireEvent.click(getByText('Increment'))
    expect(getByText('Count: 1')).toBeInTheDocument()
    })
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    ### ¿Cómo puedo hacer testing de un hook?

    Para hacer testing de un hook, puedes usar la función `renderHook` de la librería `@testing-library/react-hooks`. Esta función nos permite renderizar un hook y obtener el resultado.

    ```jsx
    import { renderHook } from '@testing-library/react-hooks'

    function useCounter() {
    const [count, setCount] = useState(0)
    const increment = () => setCount(count + 1)
    return { count, increment }
    }

    test('useCounter', () => {
    const { result } = renderHook(() => useCounter())

    expect(result.current.count).toBe(0)
    act(() => {
    result.current.increment()
    })
    expect(result.current.count).toBe(1)
    })
    ```

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es Flux?

    *Flux* es un patrón de arquitectura de aplicaciones que se basa en un unidireccional de datos. En este patrón, los datos fluyen en una sola dirección: de las vistas a los stores.

    No es específico de React y se puede usar con cualquier librería de vistas. En este patrón, los stores son los encargados de almacenar los datos de la aplicación. Los stores emiten eventos cuando los datos cambian. Las vistas se suscriben a estos eventos para actualizar los datos.

    Esta arquitectura fue creada por Facebook para manejar la complejidad de sus aplicaciones. *Redux* se basó en este patrón para crear una biblioteca de gestión de estado global.

    **[⬆ Volver a índice](#índice)**

    ---

    ### Errores Típicos en React

    #### ¿Qué quiere decir: Warning: Each child in a list should have a unique key prop?

    Es un error bastante común en React y que puede parecernos un poco extraño si estamos empezando a aprender esta tecnología. Por suerte, es bastante sencillo de solucionar.

    Básicamente, este mensaje aparece en la consola cuando estamos renderizando un listado dentro de nuestro componente, pero no le estamos indicando la propiedad "key". React usa esta propiedad para **determinar qué elemento hijo dentro de un listado ha sufrido cambios,** por lo que funciona como una especie de identificativo.

    De esta manera, React utiliza esta información para **identificar las diferencias existentes con respecto al DOM** y optimizar la renderización del listado, determinando qué elementos necesitan volverse a calcular. Esto habitualmente pasa cuando agregamos, eliminamos o cambiamos el orden de los items en una lista.

    Recomendamos revisar las siguientes secciones:

    - [¿Qué es el renderizado de listas en React?](#qué-es-el-renderizado-de-listas-en-react)

    - [¿Por qué puede ser mala práctica usar el ´index´ como key en un listado de React?](#por-qué-puede-ser-mala-práctica-usar-el-index-como-key-en-un-listado-de-react)

    **[⬆ Volver a índice](#índice)**

    ---

    #### React Hook useXXX is called conditionally. React Hooks must be called in the exact same order in every component render

    Una de las reglas de los hooks de React es que deben llamarse en el mismo orden en cada renderizado. React lo necesita para saber en qué orden se llaman los hooks y así mantener el estado de los mismos internamente. Por ello, los hooks no pueden usarse dentro de una condición `if`, ni un loop, ni tampoco dentro de una función anónima. Siempre deben estar en el nivel superior de la función.

    Por eso el siguiente código es incorrecto:

    ```jsx
    // ❌ código incorrecto por saltar las reglas de los hooks
    function Counter() {
    const [count, setCount] = useState(0)

    // de forma condicional, creamos un estado con el hook useState
    // lo que rompe la regla de los hooks
    if (count > 0) {
    const [name, setName] = useState('midu')
    }

    return

    {count} {name}

    }
    ```

    También el siguiente código sería incorrecto, aunque no lo parezca, ya que estamos usando el segundo `useState` de forma condicional (pese a no estar dentro de un `if`) ya que se ejecutará sólo cuando `count` sea diferente a `0`:

    ```jsx
    // ❌ código incorrecto por saltar las reglas de los hooks
    function Counter() {
    const [count, setCount] = useState(0)

    // si count es 0, no se ejecuta el siguiente hook useState
    // ya que salimos de la ejecución aquí
    if (count === 0) return null

    const [name, setName] = useState('midu')

    return

    {count} {name}

    }
    ```

    Ten en cuenta que si ignoras este error, es posible que tus componentes no se comporten de forma correcta y tengas comportamientos no esperados en el funcionamiento de tus componentes.

    Para arreglar este error, como hemos comentado antes, debes asegurarte de que los hooks se llaman en el mismo orden en cada renderizado. El último ejemplo quedaría así:

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)
    // movemos el hook useState antes del if
    const [name, setName] = useState('midu')

    if (count === 0) return null

    return

    {count} {name}

    }
    ```

    Recomendamos revisar las siguientes secciones:

    - [¿Cuáles son las reglas de los hooks en React?](#cuáles-son-las-reglas-de-los-hooks-en-react)

    **[⬆ Volver a índice](#índice)**

    ---

    #### Can’t perform a React state update on an unmounted component

    Este error se produce cuando intentamos actualizar el estado de un componente que ya no está montado. Esto puede ocurrir cuando el componente se desmonta antes de que se complete una petición asíncrona, por ejemplo:

    ```jsx
    function Movies () {
    const [movies, setMovies] = useState([])

    useEffect(() => {
    fetchMovies().then(() => {
    setMovies(data.results)
    })
    })

    if (!movies.length) return null

    return (

    {movies.map(movie => (

    {movie.title}


    {movie.overview}



    ))}

    )
    }
    ```

    Parece un código inofensivo, pero imagina que usamos este componente en una página. Si el usuario navega a otra página antes de que se complete la petición, el componente se desmontará y React lanzará el error, ya que intentará ejecutar el `setMovies` en un componente (Movies) que ya no está montado.

    Para evitar este error, podemos usar una variable booleana con `useRef` que nos indique si el componente está montado o no. De esta manera, podemos evitar que se ejecute el `setMovies` si el componente no está montado:

    ```jsx
    function Movies () {
    const [movies, setMovies] = useState([])
    const mounted = useRef(false)

    useEffect(() => {
    mounted.current = true

    fetchMovies().then(() => {
    if (mounted.current) {
    setMovies(data.results)
    }
    })

    return () => mounted.current = false
    })

    // ...
    }
    ```

    Esto soluciona el problema pero **no evita que se haga la petición aunque el componente ya no esté montado**. Para cancelar la petición y así ahorrar transferencia de datos, podemos abortar la petición usando la API `AbortController`:

    ```jsx
    function Movies () {
    const [movies, setMovies] = useState([])

    useEffect(() => {
    // creamos un controlador para abortar la petición
    const abortController = new AbortController()

    // pasamos el signal al fetch para que sepa que debe abortar
    fetchMovies({ signal: abortController.signal })
    .then(() => {
    setMovies(data.results)
    }).catch(error => {
    if (error.name === 'AbortError') {
    console.log('fetch aborted')
    }
    })

    return () => {
    // al desmontar el componente, abortamos la petición
    // sólo funcionará si la petición sigue en curso
    abortController.abort()
    }
    })

    // ...
    }

    // Debemos pasarle el parámetro signal al `fetch`
    // para que enlace la petición con el controlador
    const fetchMovies = ({ signal }) => {
    return fetch('https://api.themoviedb.org/3/movie/popular', {
    signal // <--- pasamos el signal
    }).then(response => response.json())
    }
    ```

    Sólo ten en cuenta la compatibilidad de `AbortController` en los navegadores. En [caniuse](https://caniuse.com/#feat=abortcontroller) puedes ver que no está soportado en Internet Explorer y versiones anteriores de Chrome 66, Safari 12.1 y Edge 16.

    **[⬆ Volver a índice](#índice)**

    ---

    #### Too many re-renders. React limits the number of renders to prevent an infinite loop

    Este error indica que algo dentro de nuestro componente está generando muchos pintados que pueden desembocar en un *loop* (bucle) infinito. Algunas de las razones por las que puede aparecer este error son las siguientes:

    1. **Llamar a una función que actualiza el estado en el renderizado del componente.**

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)

    // ❌ código incorrecto
    // no debemos actualizar el estado de manera directa
    setCount(count + 1)

    return

    {count}

    }
    ```

    Lo que sucede en este ejemplo, es que al *renderizarse* el componente, se llama a la función `setCount` para actualizar el estado. Una vez el estado es actualizado, se genera nuevamente un *render* del componente y se repite todo el proceso infinitas veces.

    Una posible solución sería:

    ```jsx
    function Counter() {
    // ✅ código correcto
    // se pasa el valor inicial deseado en el `useState`
    const [count, setCount] = useState(1)

    return

    {count}

    }
    ```

    **Llamar directamente a una función en un controlador de eventos.**

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)

    // ❌ código incorrecto
    //se ejecuta directamente la función `setCount` y provoca un renderizado infinito
    return


    Contador: {count}


    Incrementar

    }
    ```

    En este código, se está ejecutando la función `setCount` que actualiza el estado en cada renderizado del componente, lo que provoca renderizaciones infinitas.

    La manera correcta sería la siguiente:

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)

    // ✅ código correcto
    // se pasa un callback al evento `onClick`
    // esto evita que la función se ejecute en el renderizado
    return


    Contador: {count}


    setCount(count + 1)}>Incrementar

    }
    ```

    **Usar incorrectamente el Hook de `useEffect`.**

    Al ver este ejemplo:

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)

    // ❌ código incorrecto
    useEffect(() => {
    setCounter(counter + 1)
    }) // 👈️ no colocar el array de dependencias

    return

    {count}

    }
    ```

    Lo que ocurre, es que al no colocar un array de dependencias en el hook de `useEffect`, estamos provocando que el código que se encuentre dentro se ejecute en cada renderizado del componente. Al llamar al `setCounter` y actualizar el estado, obtenemos nuevamente renderizaciones infinitas.

    Para solucionarlo, podemos hacer lo siguiente:

    ```jsx
    function Counter() {
    const [count, setCount] = useState(0)

    // ✅ código correcto
    // estamos indicando que sólo queremos que el código se ejecute una vez
    useEffect(() => {
    setCounter(counter + 1)
    }, []) //colocamos un array de dependencias vacío.

    return

    {count}

    }
    ```

    Estas son solo algunas de las posibles causas que podemos encontrar cuando nos topamos con este mensaje de error en el código. Si quieres complementar esta información, te recomendamos revisar las siguientes secciones:

    - [¿Qué es el estado en React?](#qué-es-el-estado-en-react)
    - [¿Qué son los hooks?](#qué-son-los-hooks)
    - [¿Qué hace el hook useState?](#qué-hace-el-hook-usestate)
    - [¿Qué hace el hook useEffect?](#qué-hace-el-hook-useeffect)
    - [¿Cuáles son las reglas de los hooks en React?](#cuáles-son-las-reglas-de-los-hooks-en-react)

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué diferencia existe entre Shadow DOM y Virtual DOM?

    El **Shadow DOM** es una API del navegador que nos permite crear un árbol de nodos DOM independiente dentro de un elemento del DOM. Esto nos permite crear componentes que no interfieran con el resto de la aplicación. Se usa especialmente con Web Components.

    El **Virtual DOM** es una representación del DOM en memoria. Esta representación se crea cada vez que se produce un cambio en el DOM. Esto nos permite comparar el DOM actual con el DOM anterior y así determinar qué cambios se deben realizar en el DOM real. Lo usa React y otras bibliotecas para hacer el mínimo número de cambios en el DOM real.

    **[⬆ Volver a índice](#índice)**

    ---

    #### ¿Qué es el Binding?

    En React, el **Binding** se refiere a la forma en que se relaciona y sincroniza el **estado** *(state)* de un componente con su **vista** *(render)*. El estado de un componente es un objeto que contiene información que puede ser utilizada para determinar cómo se debe mostrar el componente. Existen **dos** tipos de binding en React: **One-Way Binding** y **Two-Way Binding**.

    **One-Way Binding** *(Enlace unidireccional)*:

    En React se refiere a la capacidad de un componente para actualizar su **estado** *(state)* y su **vista** *(render)* de manera automática cuando cambia el estado, pero no permitiendo que la vista actualice el estado. En otras palabras, el **one-way binding** significa que el flujo de datos es unidireccional, desde el estado hacia la vista, y no al revés.

    Por ejemplo:

    ```jsx
    import React, { useState } from 'react';

    function OneWayBindingExample() {
    const [name, setName] = useState('midu');

    return (


    Hello, {name}


    setName(e.target.value)}
    />

    );
    }

    export default OneWayBindingExample;
    ```

    *En este ejemplo, el componente tiene un estado inicial llamado **name** con el valor **midu**. La función **setName** se utiliza para actualizar el estado **name** cuando se produce un evento **onChange** en el input. Sin embargo, la **vista** (la linea que muestra **Hello, {name}**) no tiene la capacidad de actualizar el estado **name**.*

    **Two-Way Binding** *(Enlace bidireccional)*:

    Se refiere a la capacidad de un componente para actualizar su estado y su vista de manera automática tanto cuando cambia el estado como cuando se produce un evento en la vista. En otras palabras, el **Two-Way Binding** significa que el flujo de datos es bidireccional, desde el estado hacia la vista y desde la vista hacia el estado. Para lograr esto se utilizan en conjunto con los eventos, como **onChange**, para capturar la información de los inputs y actualizar el estado, *React no proporciona un mecanismo nativo para two-way binding, pero se puede lograr utilizando librerías como react-forms o formik.*

    Por ejemplo:

    ```jsx
    import React, { useState } from 'react';

    function TwoWayBindingExample() {
    const [name, setName] = useState('midu');

    return (


    Hello, {name}


    setName(e.target.value)}
    />

    );
    }

    export default TwoWayBindingExample;
    ```

    *En este ejemplo, el componente tiene un estado inicial llamado **name** con el valor **midu**. La función **setName** se utiliza para actualizar el estado **name** cuando se produce un evento **onChange** en el input, y se puede ver reflejado en el valor del input. Sin embargo, en este caso se está utilizando el atributo **value** para que el valor del input sea actualizado con el valor del estado, es decir, se está actualizando tanto el estado como el input.*

    **Por si no quedó claro:**

    En términos sencillos, el **Binding** en React puede compararse con una cafetera y una taza de café. **El estado** del componente sería la *cafetera*, y **la vista** del componente sería *la taza de café*.

    En el caso del **One-Way Binding**, la cafetera solo puede verter café en una dirección, hacia la taza de café. Esto significa que la cafetera puede llenar automáticamente la taza de café con café fresco, pero la taza de café no puede devolver automáticamente el café a la cafetera. De esta manera, **el estado** del componente *(la cafetera)* puede actualizar automáticamente **la vista** *(la taza de café)* cuando cambia, pero la **vista** no puede actualizar automáticamente el **estado**.

    En el caso del **Two-Way Binding**, la cafetera puede verter y recibir café en ambas direcciones, hacia y desde la taza de café (no sé por qué alguien necesitaría hacer algo así). Esto significa que la cafetera puede llenar y vaciar automáticamente la taza de café con café fresco. De esta manera, tanto **el estado** del componente como **la vista** pueden actualizarse automáticamente entre sí.

    Sí quieres saber más comparto el siguiente enlace:
    [How To Bind Any Component to Data in React: One-Way Binding](https://www.telerik.com/blogs/how-to-bind-any-component-data-react-one-way-binding)

    **[⬆ Volver a índice](#índice)**

    ---