Ciclo de vida de los Componente en React
Last updated
Was this helpful?
Last updated
Was this helpful?
En esta sesión veremos los distintos métodos del ciclo de vida de los componentes de React. Veremos ejemplos prácticos de cómo hacer operaciones comunes con los métodos del ciclo de vida.
Según vamos creando aplicaciones web más completas con React, necesitaremos un control más detallado de nuestros componentes. En aplicaciones con muchos componentes, resulta importante liberar recursos que usan los componentes una vez ya no se usan, por rendimiento pero también para evitar futuros errores. Por ejemplo, podemos usar componentes que se recarguen automáticamente, como puede ser una tabla que comprueba si hay puntuaciones nuevas de un partido de baloncesto.
Utilizaremos los métodos del ciclo de vida de los componentes de React para que nuestros componentes sean limpios y no creen errores evitables.
Se llama ciclo de vida al tiempo que pasa desde que un objeto se crea desde el código hasta que se elimina. En un nivel un poco más técnico, podríamos decir que desde que se carga en memoria hasta que se elimina de la memoria. Durante la vida de un componente de React, se ejecutan varios métodos, en función del momento. A estos métodos se les llama métodos del ciclo de vida. Algunos métodos del ciclo de vida que ya conocemos son el constructor()
, que se ejecuta cuando se crea el componente, y render()
, que sabemos que se ejecuta en algún momento después de crearse y cada vez que cambia el estado.
Podemos clasificar los métodos del ciclo de vida en tres tipos:
De montaje: los que se ejecutan en la fase de creación del componente.
De actualización: los que se ejecutan mientras el componente vive.
De desmontaje: los que se ejecutan antes de que el componente se destruya.
El montaje es la primera fase del ciclo de vida de un componente. Es la parte en la que se crea el componente. Sabemos que un componente de React representa un elemento del DOM y lo que contiene. En el momento en que ese elemento se pinta en el DOM, aparece visualmente en la página web, decimos que ese componente está montado. Como ya sabemos que el método render()
es el encargado de pintar el componente, podemos decirlo de otra manera: un componente se monta en el momento en que se ejecuta su render()
por primera vez.
Sin perder de vista la primera ejecución de render()
, que nos servirá de referencia, vamos a ver el resto de métodos de la fase de montaje en orden de ejecución:
constructor()
: este ya lo conocemos. Se ejecuta según se crea el componente por código y se le pasan las props
iniciales. Aquí:
inicializamos el estado
enlazamos los event handlers a la instancia con .bind(this)
render()
: otro viejo amigo. En este:
devolvemos lo que se pinta en función de props
y state
componentDidMount()
: literalmente, "el componente se ha montado". Este método se ejecuta justo después de que el componente se haya montado (pintado en pantalla). Aquí:
podemos pedir datos remotos, con fetch()
, por ejemplo
podemos "suscribir" el componente, por ejemplo, a un setInterval()
u otro código que nos dé datos de manera periódica o de tanto en tanto
Si el montaje es la primera fase del ciclo de vida de un componente, el desmontaje es la última fase del ciclo de vida del componente. Es la parte en la que se va a destruir el componente y va a dejar de mostrarse en pantalla y de existir en memoria.
Esta fase solo tiene un método: el método componentWillUnmount()
(lit. "el componente se va a desmontar"). En este método limpiaremos todo lo residual que pueda dejar nuestro componente una vez no exista. Podemos pensarlo como la contraparte de componentDidMount()
, porque será aquí donde debamos dar de baja las suscripciones que hayamos iniciado allí.
Si no limpiásemos lo residual del componente, nos aparecerán errores de partes del código que intentan acceder a un componente que ya no existe.
Como ya sabemos, mientras un componente está montado, si cambian las props
o el estado, el componente se vuelve a render
izar. Esto ocurre siempre por defecto. Sin embargo, con los métodos del ciclo de vida podemos adaptar esto a nuestras necesidades: podremos hacer operaciones en distintos puntos de la actualización o hasta impedir que el componente se re-render
ice si se dan unas condiciones.
Estos métodos son paralelos a los métodos del montaje del componente. Como tienen algunas peculiaridades, los desglosaremos en las siguientes subsecciones, pero se ejecutan en este orden:
shouldComponentUpdate()
: decide si el componente se actualiza visualmente; es decir si los dos métodos siguientes se ejecutan o no:
componentDidUpdate()
: similar a componentDidMount()
componentDidUpdate(prevProps, prevState)
se llama justo después de re-render
izar un componente por actualización de sus props
o estado. Si el componente hace peticiones que dependen de una prop
, este es buen lugar para rehacerlas, después de comprobar que efectivamente esa prop
en concreto ha cambiado.
Nota: como veremos más abajo, este método no se ejecutará si
shouldComponentUpdate()
devuelvefalse
render
izar un componenteEn la fase de actualización del ciclo de vida, tenemos el método shouldComponentUpdate(nextProps, nextState)
(lit. "¿debe el componente actualizarse?"). Este método debe devolver un booleano. Si se devuelve un booleano false
, entonces no se ejecutarán ni render()
, ni componentDidUpdate()
.
En este método podremos comparar los cambios entre las props
y el estado actuales (this.props
y this.state
) con las props
y el estado que se van a recibir (nextProps
y nextState
) para decidir si queremos que se repinte el componente o no.
Este método no se llama cuando se llama a forceUpdate()
.
Primero identificamos la fase del ciclo de vida que nos interesa. Queremos que se haga lo más pronto posible, así que será en la fase de montaje. En la fase de montaje hay varios métodos del ciclo de vida:
constructor()
render()
componentDidMount()
¿Qué método usaremos? Bien, tenemos que ver cómo obtenemos los datos, si síncrona o asíncronamente. Como vamos a recibir los datos de una llamada AJAX a un servidor, será asíncrono:
En el constructor()
solo debemos guardar datos en el estado de forma síncrona; es decir, datos que ya tenemos disponibles
En render()
solo debemos hacer operaciones con los datos que ya tenemos de las props
y el estado
En componentDidMount()
podemos hacer llamadas asíncronas y pasar un callback que guarde esos datos en el estado
El método componentDidMount()
se ejecuta después de render()
; es decir, una vez el componente ya está iniciado y pintado, listo para recibir actualizaciones al estado.
La hora con ciclo de vida
Vamos a partir del componente Clock
del ejercicio 1 de la sesión 3.7 sobre el estado. Y vamos a usar métodos del ciclo de vida para estructurar mejor el código.
PISTA: en el
constructor
no deberíamos llamar asetInterval
sino en el método de ciclo de vida adecuado
El menú dinámico
Vamos a crear un menú de opciones dinámico, es decir, que las opciones vienen de hacer una petición a un servidor. Vamos a ver paso por paso cómo hacerlo:
Creamos un componente App
que será el contenedor de la aplicación
Creamos un componente Menu
al que le pasamos por props
un array con las opciones en este formato:
Hacemos que el componente App
tenga dentro un Menu
pasándole un array que creamos "a mano", es decir, creando una variable con el array
Creamos un estado en el componente App
para almacenar las opciones de menú, que seguimos creando "a mano" en el constructor
BONUS: Creamos un botón en un nuevo componente Button
que al hacer clic vuelve a realizar una petición al servidor y actualiza el estado, por lo que el menú vuelve a pintarse. Para esto necesitamos hacer lifting para pasar la información del evento desde Button
hasta App
, que es quien mantiene el estado.
Documentación oficial de React (en inglés).
Fuente:
render()
: siempre puro y fiel,
Tomemos el ejemplo de lifting de estados de la sesión 3.10 en el que creábamos una lista de razones (). Teníamos un método fetchNewReasons()
que actualizaba el estado, y pasábamos el método por props
al componente botón. Sin embargo, teníamos que esperar a que el usuario pulsase el botón para mostrar resultados por primera vez. ¿Cómo haríamos para que el propio elemento los cargase?
Realizamos una petición al servidor en que nos devuelve un listado de opciones. La petición la hacemos con fetch
en el método del ciclo de vida que corresponda.