useEffect
useEffect
es un Hook de React que le permite sincronizar un componente con un sistema externo.
useEffect(setup, dependencies?)
- Uso
- Conexión a un sistema externo
- Envolver los efectos en Hooks personalizados
- Controlar un widget que no sea de React
- Obtención de datos con Efectos
- Especificación de dependencias reactivas
- Actualización del estado basado en el estado anterior de un efecto
- Eliminación de dependencias de objetos innecesarios
- Eliminación de dependencias de funciones innecesarias
- Lectura de los últimos accesorios y el estado de un Efecto
- Mostrar contenidos diferentes en el servidor y en el cliente
- Referencia
- Solución de problemas
- Mi efecto se ejecuta dos veces cuando el componente se monta
- Mi efecto se ejecuta después de cada re-renderización
- Mi efecto se repite en un ciclo infinito
- Mi lógica de limpieza se ejecuta a pesar de que mi componente no se ha desmontado
- Mi efecto hace algo visual, y veo un parpadeo antes de que se ejecute
Uso
Conexión a un sistema externo
A veces, un componente puede necesitar permanecer conectado a la red, a alguna API del navegador, o a una librería de terceros, mientras se muestra en la página. Estos sistemas no están controlados por React, por lo que se denominan externos.
Para conectar su componente a algún sistema externo, declare useEffect
en el nivel superior de su componente:
import { useEffect } from 'react';
import { createConnection } from './chat.js';
function ChatRoom({ roomId }) {
const [serverUrl, setServerUrl] = useState('https://localhost:1234');
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => {
connection.disconnect();
};
}, [serverUrl, roomId]);
// ...
}
Tienes que pasar dos argumentos a useEffect
:
- Una función de configuración con código de configuración que se conecta a ese sistema.
- Debería devolver una función de limpieza con código de limpieza que se desconecta de ese sistema.
- Una Lista de dependecias incluyendo cada valor de su componente utilizado dentro de esas funciones.
React llama a sus funciones de configuración y limpieza siempre que es necesario, lo que puede ocurrir varias veces:
- Tú código de configuración se ejecuta cuando su componente se añade a la página (se monta).
- Después de cada re-renderización de su componente donde las dependencias han cambiado:
- Primero, tu código de limpieza se ejecuta con las antiguas props y estados.
- Entonces, tu código de configuración se ejcutará con las nuevas props y estados.
- Tú código de limpieza se ejecutara una última vez después de que tú componente sea eliminado de la página (se desmonta).
Vamos a mostrar esta secuencia para el ejemplo anterior.
Cuando el componente ChatRoom
se añade a la página, se conectará a la sala de conversación con el serverUrl
y roomId
. Si cualquiera de los dos serverUrl
o roomId
cambian como resultado de una re-renderización (digamos, si el usuario elige una sala de chat diferente en un desplegable), tú Efecto se desconectará de la sala anterior, y se conectara a la siguiente. Cuando el componente ChatRoom
sea finalmente eliminado de la página, su efecto se desconectará por última vez.
Para ayudarte a encontrar errores, en el desarollador de React ejecuta la configuración y la limpiezauna vez más antes de la configuración real. Se trata de una prueba de estrés que verifica que la lógica de tu efecto se implemnta correctamente. Si esto causa problemas visibles, tu función de limpieza está perdiendo algo de lógica. La función de limpieza debe detener o deshacer lo que la función de configuración esta haciendo. La regla general es que el usuario no debería ser capaz de distinguir entre la configuración que se llama una vez (como en producción) y una secuencia de configuración → limpieza → configuración (como en desarrollo). Vea las soluciones comunes.
Intenta escribir cada efecto como un proceso independiente y sólo piensa en un único ciclo de montaje/limpieza a la vez. No debería importar si tu componente se está montando, actualizando o desmontando. Cuando tu lógica de limpieza “refleja” correctamente la lógica de configuración, tu Efecto será capaz de ejecutar la configuración y limpieza tantas veces como sea necesario.
Ejemplo 1 de 5: Conexión a un sevidor de chat
En este ejemplo, el componente ChatRoom
utiliza un Effecto para permancer conectado a un sistema externo definido en chat.js
. Pulsa “Abrir chat” para que aparezca el componente ChatRoom
. Este sandbox se ejecuta en modo de desarrollo, por lo que hay un ciclo extra de conexión y desconexión, como se explica aquí. Prueba a cambiar el roomId
y serverUrl
usando el desplegable y la entrada, y observa como el efecto se reconecta con el chat. Pulsa “Cerrar chat” para ver cómo el Efecto se desconectara por última vez.
import { useState, useEffect } from 'react'; import { createConnection } from './chat.js'; function ChatRoom({ roomId }) { const [serverUrl, setServerUrl] = useState('https://localhost:1234'); useEffect(() => { const connection = createConnection(serverUrl, roomId); connection.connect(); return () => { connection.disconnect(); }; }, [roomId, serverUrl]); return ( <> <label> URL del servidor:{' '} <input value={serverUrl} onChange={e => setServerUrl(e.target.value)} /> </label> <h1>Bienvenido al sitio {roomId}!</h1> </> ); } export default function App() { const [roomId, setRoomId] = useState('general'); const [show, setShow] = useState(false); return ( <> <label> Elija el sitio de chat:{' '} <select value={roomId} onChange={e => setRoomId(e.target.value)} > <option value="general">general</option> <option value="travel">viaje</option> <option value="music">música</option> </select> </label> <button onClick={() => setShow(!show)}> {show ? 'Cerrar chat' : 'Abrir chat'} </button> {show && <hr />} {show && <ChatRoom roomId={roomId} />} </> ); }
Envolver los efectos en Hooks personalizados
los efectos son una “escotilla de escape”: los usas cuando necesitas “salirte React” y cuando no hay una mejor solución. Si te encueentras a menudo con la necesidad de escribir manualmente los efectos, suele ser una señal de que necesitas extraer algunos Hooks personalizados para los comportamientos comunes de los que dependen tus componentes.
Por ejemplo, este Hook personalizado useChatRoom
“esconde” la lógica de su efecto detrás de una API más declarativa:
function useChatRoom({ serverUrl, roomId }) {
useEffect(() => {
const options = {
serverUrl: serverUrl,
roomId: roomId
};
const connection = createConnection(options);
connection.connect();
return () => connection.disconnect();
}, [roomId, serverUrl]);
}
Entonces puedes usarlo desde cualquier componente como este:
function ChatRoom({ roomId }) {
const [serverUrl, setServerUrl] = useState('https://localhost:1234');
useChatRoom({
roomId: roomId,
serverUrl: serverUrl
});
// ...
También hay muchos exelentes Hooks personalizados para cada propósito desponibles en el entorno de React
Más información sobre cómo envolver los efectos en Hooks personalizados.
Ejemplo 1 de 3: Hook personalizado useChatRoom
Este ejemplo es idéntico a uno de los anteriores ejemplos, pero la lógica se extrae de un Hook personalizado.
import { useState } from 'react'; import { useChatRoom } from './useChatRoom.js'; function ChatRoom({ roomId }) { const [serverUrl, setServerUrl] = useState('https://localhost:1234'); useChatRoom({ roomId: roomId, serverUrl: serverUrl }); return ( <> <label> URL del Servidor:{' '} <input value={serverUrl} onChange={e => setServerUrl(e.target.value)} /> </label> <h1>Bienvenido al sitio {roomId}!</h1> </> ); } export default function App() { const [roomId, setRoomId] = useState('general'); const [show, setShow] = useState(false); return ( <> <label> Elige el sitio de chat::{' '} <select value={roomId} onChange={e => setRoomId(e.target.value)} > <option value="general">general</option> <option value="travel">viaje</option> <option value="music">música</option> </select> </label> <button onClick={() => setShow(!show)}> {show ? 'Cerrar chat' : 'Abrir chat'} </button> {show && <hr />} {show && <ChatRoom roomId={roomId} />} </> ); }
Controlar un widget que no sea de React
A veces, quieres mantener un sistema externo sincronizado con alguna prop o estado de tu componente.
Por ejemplo, si tienes un widget de mapa de terceros o un componente reproductor de vídeo escrito sin React, puedes usar un Effect para llamar a los métodos en él que hagan que su estado coincida con el estado actual de tu componente de React. Este efecto crea una instancia de la clase MapWidget
definida en map-widget.js
. Cuando cambias la propiedad zoomLevel
del componente Map
, el efecto llama a setZoom()
en la instacia de la clase para mantenerla sincronizada:
import { useRef, useEffect } from 'react'; import { MapWidget } from './map-widget.js'; export default function Map({ zoomLevel }) { const containerRef = useRef(null); const mapRef = useRef(null); useEffect(() => { if (mapRef.current === null) { mapRef.current = new MapWidget(containerRef.current); } const map = mapRef.current; map.setZoom(zoomLevel); }, [zoomLevel]); return ( <div style={{ width: 200, height: 200 }} ref={containerRef} /> ); }
En este ejemplo,no se necesita una función de limpieza porque la clase MapWidget
solo gestiona el nodo DOM que se le pasó. Después de que el componente de React Map
se elimine del árbol, tanto el nodo DOM como la instancia de la clase MapWidget
serán recogidos automáticamente por el motor JavaScript del navegador.
Obtención de datos con Efectos
Puede utilizar un efecto para obtener datos para tu componente. Ten encuenta que si utilizas un framework, usar el mecanismo de datos de tu framework será mucho más eficiente que escribir los efectos manualmente.
Si quieres obtener datos e un Efecto manualmete, tu código podría ser así:
import { useState, useEffect } from 'react';
import { fetchBio } from './api.js';
export default function Page() {
const [person, setPerson] = useState('Alice');
const [bio, setBio] = useState(null);
useEffect(() => {
let ignore = false;
setBio(null);
fetchBio(person).then(result => {
if (!ignore) {
setBio(result);
}
});
return () => {
ignore = true;
};
}, [person]);
// ...
Observe la variable ignore
que se inicializa en false
, y se establece true
durante la limpieza. Esto asegura quetu código no sufra de “condiciones de carrera”:las respuestas de la red pueden llegar en un orden diferente al que las enviaste.
import { useState, useEffect } from 'react'; import { fetchBio } from './api.js'; export default function Page() { const [person, setPerson] = useState('Alice'); const [bio, setBio] = useState(null); useEffect(() => { let ignore = false; setBio(null); fetchBio(person).then(result => { if (!ignore) { setBio(result); } }); return () => { ignore = true; } }, [person]); return ( <> <select value={person} onChange={e => { setPerson(e.target.value); }}> <option value="Alice">Alice</option> <option value="Bob">Bob</option> <option value="Taylor">Taylor</option> </select> <hr /> <p><i>{bio ?? 'Cargando...'}</i></p> </> ); }
También puede reescribir usando la sintaxis async
/ await
, pero todavía necesitas proporcionar una función de limpieza:
import { useState, useEffect } from 'react'; import { fetchBio } from './api.js'; export default function Page() { const [person, setPerson] = useState('Alice'); const [bio, setBio] = useState(null); useEffect(() => { async function startFetching() { setBio(null); const result = await fetchBio(person); if (!ignore) { setBio(result); } } let ignore = false; startFetching(); return () => { ignore = true; } }, [person]); return ( <> <select value={person} onChange={e => { setPerson(e.target.value); }}> <option value="Alice">Alice</option> <option value="Bob">Bob</option> <option value="Taylor">Taylor</option> </select> <hr /> <p><i>{bio ?? 'Cargando...'}</i></p> </> ); }
Escribir la obtención de datos directamente en los Effects se vuelve repetitivo y dificulta la adición de optimizadores como el almacenamiento en caché y el renderizado del servidor más adelante. Es más fácil utilizar un Hook personalizado, ya sea propio o hecho por la comunidad.
Deep Dive
What are good alternatives to data fetching in Effects?
What are good alternatives to data fetching in Effects?
Escribir llamadas fetch
dentro de Efectos es una forma porpular de obtener datos, especialmente en aplicaciones totalmente del lado del cliente. Sin embargo, este es un enfoque muy manual y tiene importantes desventajas:
- Los efectos no se ejecutan en el servidor. Esto significa que el HTML inicia renderizado en el servidor sólo incluirá un estado de carga sin datos. El ordenador del cliente tendrá que descargar todo el JavaScript y renderizar su aplicación sólo para descubrir que ahora necesita cargar los datos. Esto no es muy eficiente.
- La obtención de datos directamente en Efectos facilita la creación de “cascadas de red”. Se renderiza el componente principal, se obtienen algunos componentes secundariosy, a continuación, éstos comienzan a obtener sus datos. Si la red no es muy rápida, esto es significativamente más lento que obtener todos los datos en paralelo.
- La obtención de datos directamente en Efectos suele significar que no se precargan ni se almacenan en caché los datos. Por ejemplo, si el componente se desmonta y se vuelve a montar, tendría que recuperar los datos de nuevo.
- No es muy ergonómico. Hay un poco de código repetitivo involucrado en la escritura de las llamadas
fetch
de una manera que no sufre de errores como las condiciones de carrera.
Esta lista de inconvenientes no es específica de React. Se aplica a la obtención de datos en el montaje con cualquier biblioteca. Al igual que con el enrutamiento, la obtención de datos no es trivial para hacerlo bien, por lo que recomendamos los siguientes enfoques:
- Si usas un framework, utiliza su mecanismo de obtención de datos integrado. Los frameworks modernos de React han integrado mecanismos de obtención de datos que son eficientes y no sufren los inconvenientes anteriores.
- De lo contrario, considere la posibilidad de utilizar o construir una caché del lado del cliente. Las soluciones populares de código abierto incluyen React Query, useSWR, y React Router 6.4+. También puedes crear tu propia solución, en cuyo caso se utilizaría Effects bajo el capó, pero también se añadiría lógica para deduplicar las peticiones, almacenar en caché las respuestas y evitar las cascadas de red (precargando los datos o elevando los requisitos de datos a las rutas).
Puede seguir obteniendo datos directamente en Effects si ninguno de estos enfoques le conviene.
Especificación de dependencias reactivas
Observa que no puedes “elegir” las dependencias de tu Efecto. Cada valor reactivo utilizado por el código de tu efecto debe ser declarado como una dependencia. La lista de dependencias de tu efecto está determinada por el código que lo rodea:
function ChatRoom({ roomId }) { // Este es un valor reactivo
const [serverUrl, setServerUrl] = useState('https://localhost:1234'); // Este es también un valor reactivo
useEffect(() => {
const connection = createConnection(serverUrl, roomId); // Este efecto lee estos valores reactivos
connection.connect();
return () => connection.disconnect();
}, [serverUrl, roomId]); // ✅ Así que debes especificarlos como dependencias de tu Efecto
// ...
}
Si el serverUrl
o el roomId
cambian, tu efecto se reconectará al chat usando los nuevos valores.
Los valores Reactivos incluyen props y todas las variables y funciones declaradas directamente dentro de su componente. Como roomId
y serverUrl
son valores reactivos, no puedes eliminarlos de la lista de dependencias. Si intentas omitirlos y tu linter está correctamente configurado para React, el linter lo marcará como un error que debes corregir:
const serverUrl = 'https://localhost:1234';
function ChatRoom({ roomId }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // 🔴 React Hook useEffect tiene dependencias faltantes: 'roomId' y 'serverUrl'
// ...
}
Para eliminar una dependencia, tienes que “demostrar” al linter que no necesita ser una dependencia. Por ejemplo, puedes mover serverUrl
fuera de tu componente para demostrar que no es reactivo y que no cambiará en las re-renderizaciones:
const serverUrl = 'https://localhost:1234'; // Ya no es un valor reactivo
function ChatRoom({ roomId }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [roomId]); // ✅ Todas las dependencias declaradas
// ...
}
Ahora que serverUrl
no es un valor reactivo (y no puede cambiar en una renderización), no necesita ser una dependencia. Si el código de tu efecto no utiliza ningún valor reactivo, su lista de dependencias debería estar vacía ([]
):
const serverUrl = 'https://localhost:1234'; // Ya no es un valor reactivo
const roomId = 'music'; // Ya no es un valor reactivo
function ChatRoom() {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // ✅ Todas las dependencias declaradas
// ...
}
Un efecto con dependencias vacías no se vuelve a ejecutar cuando cambian los accesorios o el estado del componente.
Ejemplo 1 de 3: Pasar un array de relaciones
Si especifica las dependencias, su Efecto se ejecuta después de la renderización inicial y después de las re-renderizaciones con las dependencias cambiadas.
useEffect(() => {
// ...
}, [a, b]); // Se ejecuta de nuevo si a o b son diferentes
En el siguiente ejemplo, serverUrl
y roomId
son valores reactivos, por lo que ambos deben ser especificados como dependencias. Como resultado, la selección de un sitio diferente en el menú desplegable o la edición de la entrada de la URL del servidor hace que el chat se vuelva a conectar. Sin embargo, dado que message
no se utiliza en el efecto (y por tanto no es una dependencia), la edición del mensaje no reconecta el chat.
import { useState, useEffect } from 'react'; import { createConnection } from './chat.js'; function ChatRoom({ roomId }) { const [serverUrl, setServerUrl] = useState('https://localhost:1234'); const [message, setMessage] = useState(''); useEffect(() => { const connection = createConnection(serverUrl, roomId); connection.connect(); return () => { connection.disconnect(); }; }, [serverUrl, roomId]); return ( <> <label> URL del servidor:{' '} <input value={serverUrl} onChange={e => setServerUrl(e.target.value)} /> </label> <h1>¡Bienvenido al sitio {roomId}!</h1> <label> Tu mensaje:{' '} <input value={message} onChange={e => setMessage(e.target.value)} /> </label> </> ); } export default function App() { const [show, setShow] = useState(false); const [roomId, setRoomId] = useState('general'); return ( <> <label> Elija el sitio de chat:{' '} <select value={roomId} onChange={e => setRoomId(e.target.value)} > <option value="general">general</option> <option value="travel">viaje</option> <option value="music">música</option> </select> <button onClick={() => setShow(!show)}> {show ? 'Cerrar chat' : 'Abrir chat'} </button> </label> {show && <hr />} {show && <ChatRoom />} </> ); }
Actualización del estado basado en el estado anterior de un efecto
Cuando quieras actualizar el estado basándote en el estado anterior de un Efecto, puedes encontrarte con un problema:
function Counter() {
const [count, setCount] = useState(0);
useEffect(() => {
const intervalId = setInterval(() => {
setCount(count + 1); // Quieres incrementar el contador cada segundo...
}, 1000)
return () => clearInterval(intervalId);
}, [count]); // 🚩 ... pero al especificar `count` como dependencia siempre se restablece el intervalo.
// ...
}
Como count
es un valor reactivo, debe ser especificado en la lista de dependencias. Sin embargo, eso hace que el Efecto se limpie y se configure de nuevo cada vez que count
cambia. Esto no es lo ideal.
Para solucionar esto, pasa el actualizador de estado c => c + 1
a setCount
:
import { useState, useEffect } from 'react'; export default function Counter() { const [count, setCount] = useState(0); useEffect(() => { const intervalId = setInterval(() => { setCount(c => c + 1); // ✅ Pasar un actualizador de estado }, 1000); return () => clearInterval(intervalId); }, []); // ✅ Ahora count no es una dependencia return <h1>{count}</h1>; }
Ahora que pasas c => c + 1
en lugar de count + 1
, tu Efecto ya no necesita depender de un count
. Como resultado de esta corrección, no tendrá que limpiar y configurar el intervalo de nuevo cada vez que el recuento cambia.
Eliminación de dependencias de objetos innecesarios
Si tu Efecto depende de un objeto o de una función creada durante el renderizado, puede que se ejecute con más frecuencia de la necesaria. Por ejemplo, este Efecto se reconecta después de cada renderización porque el objeto options
esdiferente para cada renderización:
const serverUrl = 'https://localhost:1234';
function ChatRoom({ roomId }) {
const [message, setMessage] = useState('');
const options = { // 🚩 Este objeto se crea desde cero en cada re-renderización
serverUrl: serverUrl,
roomId: roomId
};
useEffect(() => {
const connection = createConnection(options); // Se usa dentro del Efecto
connection.connect();
return () => connection.disconnect();
}, [options]); // 🚩 Como resultado, estas dependencias son siempre diferentes en una renderización
// ...
Evite utilizar como dependencia un objeto creado durante la renderización. En su lugar, cree el objeto dentro del Efecto:
import { useState, useEffect } from 'react'; import { createConnection } from './chat.js'; const serverUrl = 'https://localhost:1234'; function ChatRoom({ roomId }) { const [message, setMessage] = useState(''); useEffect(() => { const options = { serverUrl: serverUrl, roomId: roomId }; const connection = createConnection(options); connection.connect(); return () => connection.disconnect(); }, [roomId]); return ( <> <h1>¡Bienvenido al sitio {roomId}!</h1> <input value={message} onChange={e => setMessage(e.target.value)} /> </> ); } export default function App() { const [roomId, setRoomId] = useState('general'); return ( <> <label> Choose the chat room:{' '} <select value={roomId} onChange={e => setRoomId(e.target.value)} > <option value="general">general</option> <option value="travel">viaje</option> <option value="music">música</option> </select> </label> <hr /> <ChatRoom roomId={roomId} /> </> ); }
Ahora que creas el objeto options
dentro del Efecto, el propio Efecto sólo depende del stringroomId
string.
Con este arreglo, escribir en la entrada no reconecta el chat. A diferencia de un objeto que se vuelve a crear, un string como roomId
no cambia a menos que la establezcas con otro valor. Más información sobre la eliminación de dependencias.
Eliminación de dependencias de funciones innecesarias
Si tu Efecto depende de un objeto o de una función creada durante el renderizado, puede que se ejecute con más frecuencia de la necesaria. Por ejemplo, este Efecto se reconecta después de cada renderización porque el objeto options
esdiferente para cada renderización:
function ChatRoom({ roomId }) {
const [message, setMessage] = useState('');
function createOptions() { // 🚩 Esta función se crea desde cero en cada renderización
return {
serverUrl: serverUrl,
roomId: roomId
};
}
useEffect(() => {
const options = createOptions(); // Se usa dentro del Efecto
const connection = createConnection();
connection.connect();
return () => connection.disconnect();
}, [createOptions]); // 🚩 Como resultado, estas dependencias son siempre diferentes en una renderización
// ...
Por sí mismo, crear una función desde cero en cada renderización no es un problema. No es necesario optimizarla. Sin embargo, si lo usas como una dependencia de tu Efecto, hará que tu Efecto se vuelva a ejecutar después de cada re-renderización.
Evite utilizar como dependencia una función creada durante el renderizado. En su lugar, declárela dentro del Efecto:
import { useState, useEffect } from 'react'; import { createConnection } from './chat.js'; const serverUrl = 'https://localhost:1234'; function ChatRoom({ roomId }) { const [message, setMessage] = useState(''); useEffect(() => { function createOptions() { return { serverUrl: serverUrl, roomId: roomId }; } const options = createOptions(); const connection = createConnection(options); connection.connect(); return () => connection.disconnect(); }, [roomId]); return ( <> <h1>¡Bienvenido al sitio {roomId}!</h1> <input value={message} onChange={e => setMessage(e.target.value)} /> </> ); } export default function App() { const [roomId, setRoomId] = useState('general'); return ( <> <label> Elige el sitio de chat:{' '} <select value={roomId} onChange={e => setRoomId(e.target.value)} > <option value="general">general</option> <option value="travel">viaje</option> <option value="music">música</option> </select> </label> <hr /> <ChatRoom roomId={roomId} /> </> ); }
Ahora que se define la función createOptions
function inside the Effect, dentro del Efecto, el Efecto mismo sólo depende del string roomId
. Con este arreglo, escribir en la entrada no reconecta el chat. A diferencia de una función que se vuelve a crear, una cadena como roomId
no cambia a menos que la establezcas con otro valor. Lea más sobre la eliminación de dependencias.
Lectura de los últimos accesorios y el estado de un Efecto
Por defecto, cuando lees un valor reactivo de un Efecto, tienes que añadirlo como una dependencia. Esto asegura que tu Efecto “reacciona” a cada cambio de ese valor. Para la mayoría de las dependencias, ese es el comportamiento que quieres.
Sin embargo, a veces querrá leer las últimas props y estados de un efecto sin “reaccionar” a ellos. Por ejemplo, imagínese que quiere registrar el número de artículos del carrito de la compra en cada visita a la página:
function Page({ url, shoppingCart }) {
useEffect(() => {
logVisit(url, shoppingCart.length);
}, [url, shoppingCart]); // ✅ Todas las dependencias declaradas
// ...
}
¿Qué pasa si quieres registrar una nueva visita a la página después de cada cambio de url
, pero no si sólo cambia el shoppingCart
? No puedes excluir shoppingCart
de las dependencias sin romper las reglas de reactividad. Sin embargo, puedes expresar que no quieres que una parte de código “reaccione” a los cambios aunque sea llamado desde dentro de un Efecto. Para hacer esto, declare un Event function con el Hook useEvent
, y mueva el código que lee shoppingCart
dentro de ella:
function Page({ url, shoppingCart }) {
const onVisit = useEvent(visitedUrl => {
logVisit(visitedUrl, shoppingCart.length)
});
useEffect(() => {
onVisit(url);
}, [url]); // ✅ Todas las dependencias declaradas
// ...
}
Los Event functions no son reactivas y no necesitan ser especificadas como dependencias de tu Efecto. Esto es lo que te permite poner código no reactivo (donde puedes leer el último valor de algunas props y estados) dentro de ellas. Por ejemplo, al leer shoppingCart
dentro de onVisit
, te aseguras de que shoppingCart
no vuelva a ejecutar tu efecto.
Lea más sobre cómo las funciones de evento le permiten separar el código reactivo del no reactivo.
Mostrar contenidos diferentes en el servidor y en el cliente
Si tu aplicación utiliza el renderizado del servidor (ya sea directamente o a través de unframework), tu componente se renderizará en dos entornos diferentes. En el servidor, se renderizará para producir el HTML inicial. En el cliente, React ejecutará de nuevo el código de renderizado para poder adjuntar tus controladores de eventos a ese HTML. Por eso, para que la integración funcione, tu salida inicial de renderizado debe ser idéntica en el cliente y en el servidor.
En raras ocasiones, es posible que necesites mostrar un contenido diferente en el cliente. Por ejemplo, si su aplicación lee algunos datos del localStorage
, no puede hacerlo en el servidor. Así es como típicamente se implementaría esto:
function MyComponent() {
const [didMount, setDidMount] = useState(false);
useEffect(() => {
setDidMount(true);
}, []);
if (didMount) {
// ... devolver JSX sólo para clientes ...
} else {
// ... devolver el JSX inicial ...
}
}
Mientras se carga la aplicación, el usuario verá la salida de renderización inicial. Luego, cuando esté cargada e integrada, su efecto se ejecutará y establecerá didMount
a true
, disparando una re-renderización. Esto cambiará a la salida de renderización sólo para el cliente. Ten en cuenta que los Efectos no se ejecutan en el servidor, por eso didMount
era false
falso durante el renderizado inicial del servidor.
Utilice este patrón con moderación. Ten en cuenta que los usuarios con una conexión lenta verán el contenido inicial durante bastante tiempo - potencialmente, muchos segundos - por lo que no querrás hacer cambios bruscos en la apariencia de tu componente. En muchos casos, puedes evitar la necesidad de esto mostrando condicionalmente diferentes cosas con CSS.
Referencia
useEffect(¿configuración, dependencias?)
Declare un efecto con useEffect
en el nivel superior de su componente:
import { useEffect } from 'react';
import { createConnection } from './chat.js';
function ChatRoom({ roomId }) {
const [serverUrl, setServerUrl] = useState('https://localhost:1234');
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => {
connection.disconnect();
};
}, [serverUrl, roomId]);
// ...
}
Parámetros
-
configuración
: La función con la lógica de su Efecto. Tu función de configuración también puede devolver opcionalmente una función de limpieza. Cuando tu componente se añade por primera vez al DOM, React ejecutará tu función de configuración. Después de cada renderización con dependencias cambiadas, React ejecutará primero la función de limpieza (si la proporcionaste) con los valores antiguos, y luego ejecutará tu función de configuración con los nuevos valores. Después de que tu componente sea eliminado del DOM, React ejecutará tu función de limpieza una última vez. -
dependencias
opcionales: La lista de todos los valores reactivos referenciados dentro del código deconfiguración
. Los valores reactivos incluyen props, estados,y todas las variables y funciones declaradas directamente dentro del cuerpo de tu componente. Si tu linter está configurado para React, verificará que cada valor reactivo esté correctamente especificado como una dependencia. La lista de dependencias debe tener un número constante de elementos y estar escrita en línea como[dep1, dep2, dep3]
. React comparará cada dependencia con su valor anterior utilizando el algoritmo de comparaciónObject.is
. Si no se especifican las dependencias en absoluto, su efecto se volverá a ejecutar después de cada renderización del componente. Mira la diferencia entre pasar un array de dependencias, un array vacío y ninguna dependencia.
Retornos
useEffect
retorna undefined
.
Advertencias
-
useEffect
es un Hook, por lo que sólo puedes llamarlo en el nivel superior de tu componente o en tus propios Hooks. No puedes llamarlo dentro de bucles o condiciones. Si lo necesitas, extrae un nuevo componente y mueve el estado a él. -
Si no estás tratando de sincronizar con algún sistema externo, probablemente no necesites un Efecto.
-
Cuando el modo estricto está activado, React ejecutará un extra ciclo extra de configuración+limpieza de desarrollo antes de la primera configuración real. Esta es una prueba de estrés que asegura que tu lógica de limpieza ” refleje” tu lógica de configuración y que detenga o deshaga cualquier cosa que la configuración esté haciendo. Si esto causa un problema, necesitas implementar la función de limpieza.
-
Si algunas de sus dependencias son objetos o funciones definidas dentro del componente, existe el riesgo de que provoquen que el Efecto se reejecuta más veces de las necesarias. Para solucionar esto, elimina las dependencias innecesarias de object y function innecesarias. También puedes extraer las actualizaciones de estados y la lógica no reactiva fuera de tu Efecto.
-
If your Effect wasn’t caused by an interaction (like a click), React will let the browser paint the updated screen first before running your Effect. If your Effect is doing something visual (for example, positioning a tooltip), and the delay is noticeable (for example, it flickers), you need to replace
useEffect
withuseLayoutEffect
. -
Los efectos sólo se ejecutan en el cliente. No se ejecutan durante el renderizado del servidor.
Solución de problemas
Mi efecto se ejecuta dos veces cuando el componente se monta
Cuando el modo estricto está activado, en el desarrollo, React ejecuta la configuración y la limpieza una vez más antes de la configuración real.
Esta es una prueba de estrés que verifica que la lógica de su efecto se implementa correctamente. Si esto causa problemas visibles, su función de limpieza está perdiendo alguna lógica. La función de limpieza debe detener o deshacer lo que la función de configuración estaba haciendo. La regla general es que el usuario no debería ser capaz de distinguir entre la configuración que se llama una vez (como en producción) y una secuencia de configuración → limpieza → configuración (como en desarrollo).
Lea más sobre cómo esto ayuda a encontrar errores y cómo arreglar su lógica.
Mi efecto se ejecuta después de cada re-renderización
En primer lugar, comprueba que no has olvidado especificar el array de dependencias:
useEffect(() => {
// ...
}); // 🚩 No hay array de dependencias: ¡se vuelve a ejecutar después de cada renderización!
Si has especificado el array de dependencias pero tu Efecto aún se reejecuta en un bucle, es porque una de tus dependencias es diferente en cada re-renderización.
Puedes depurar este problema registrando manualmente tus dependencias en la consola:
useEffect(() => {
// ..
}, [serverUrl, roomId]);
console.log([serverUrl, roomId]);
A continuación, puede hacer clic con el botón derecho del ratón en las matrices de las diferentes renderizaciones en la consola y seleccionar “Guardar como variable global” para ambas. Suponiendo que la primera se guardó como temp1
y la segunda se guardó como temp2
, entonces puedes usar la consola del navegador para comprobar si cada dependencia en ambos array es la misma:
Object.is(temp1[0], temp2[0]); // ¿La primera dependencia es la misma entre los array?
Object.is(temp1[1], temp2[1]); // ¿La segunda dependencia es la misma entre los arrays?
Object.is(temp1[2], temp2[2]); // ... y así sucesivamente para cada dependencia ...
Cuando encuentres la dependencia que es diferente en cada renderización, normalmente puedes arreglarlo de una de estas maneras:
- Actualización del estado basado en el estado anterior de un efecto
- Eliminación de dependencias de objetos innecesarias
- Eliminación de dependencias de funciones innecesarias
- Lectura de los últimos accesorios y estados de un efecto
Como último recurso (si estos métodos no ayudan), envuelva su creación con useMemo
o useCallback
(para funciones).
Mi efecto se repite en un ciclo infinito
Si tu Efecto funciona en un ciclo infinito, estas dos cosas deben ser ciertas:
- Tu efecto está actualizando algún estado.
- Ese estado lleva a una re-renderización, lo que hace que las dependencias del Efecto cambien.
Antes de empezar a solucionar el problema, pregúntate si tu efecto se está conectando a algún sistema externo (como el DOM, la red, un widget de terceros, etc.). ¿Por qué tu efecto necesita establecer un estado? ¿Sincroniza algún estado con ese sistema externo? ¿O estás intentando gestionar el flujo de datos de tu aplicación con él?
Si no hay un sistema externo, considere si la eliminación del Efecto por completo simplificaría su lógica.
Si realmente estás sincronizando con algún sistema externo, piensa por qué y bajo qué condiciones tu Efecto debe actualizar el estado. ¿Ha cambiado algo que afecta a la salida visual de tu componente? Si necesitas hacer un seguimiento de algunos datos que no son utilizados por la renderización, un ref (que no desencadena la re-renderización) podría ser más apropiada. Compruebe que su efecto no actualiza el estado (y no provoca la re-renderización) más de lo necesario.
Por último, si tu efecto está actualizando el estado en el momento adecuado, pero sigue habiendo un bucle, es porque esa actualización de estado hace que cambie una de las dependencias de tu efecto. Lee cómo depurar y resolver los cambios de dependencias.
Mi lógica de limpieza se ejecuta a pesar de que mi componente no se ha desmontado
La función de limpieza se ejecuta no sólo durante el desmontaje, sino antes de cada renderización con dependencias cambiadas. Además, en el desarrollo, React ejecuta setup+cleanup una vez más inmediatamente después de montar el componente.
Si tienes código de limpieza sin el correspondiente código de configuración, suele ser un error de código:
useEffect(() => {
// 🔴 Avoid: Lógica de limpieza sin la correspondiente lógica de configuración
return () => {
doSomething();
};
}, []);
Su lógica de limpieza debe ser “simétrica” a la lógica de configuración, y debe detener o deshacer lo que hizo la configuración:
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => {
connection.disconnect();
};
}, [serverUrl, roomId]);
Aprenda cómo el ciclo de vida del efecto es diferente del ciclo de vida del componente.
Mi efecto hace algo visual, y veo un parpadeo antes de que se ejecute
Si tu efecto debe bloquear el navegador para que no pinte la pantalla, sustituye useEffect
por useLayoutEffect
. Ten en cuenta que esto no debería ser necesario para la gran mayoría de los Efectos. Sólo lo necesitarás si es crucial ejecutar tu Efecto antes de que el navegador pinte: por ejemplo, para medir y posicionar un tooltip antes de que el usuario lo vea por primera vez.