Code splitting e lazy loading no React

Matheus Castiglioni
CollabCode
Published in
6 min readNov 30, 2018

--

A maioria dos aplicativos feitos com a biblioteca React, terão seus arquivos empacotados, esse empacotamento pode ser feito com ferramentas de empacotação, por exemplo: Webpack ou Parcel (além de outras). Realizar o cmpacotamento de uma aplicação é o processo onde todos os arquivos importados e usados no projeto serão mergeados para um único arquivo. Esse único arquivo é onde todo o projeto foi empacotado e mesclado, ele é o responsável por carregar a aplicação inteira de uma vez e deixá-la funcionando.

Por exemplo:

A aplicação para exemplo do post não tem nada demais, apenas um sistema de rotas, onde:

  • /: Renderiza o HomeComponent.
  • /a: Renderiza o AComponent.
  • /b: Renderiza o BComponent.
  • /c: Renderiza o CComponent.

As rotas não estão sendo visíveis, pois, preferi gravar apenas a aplicação em funcionamento e não o navegador de forma completa.

Se olharmos na aba network do navegador (no caso do Chrome), podemos ver que tudo está sendo baixado de uma única vez:

Mas, será que podemos melhorar nossa aplicação?

Sim, podemos melhorar o desempenho e performance da mesma, assunto do tópico á seguir.

Code Splitting

Uma técnica muito famosa para resolver esse tipo de problema é chamada de Code Splitting (separação de código), como funciona essa técnica? Basicamente, pegamos nosso único arquivo .js que foi empacotado e o dividimos em outros pequenos arquivos (geralmente um para cada componente). Dessa maneira, não teremos apenas um .js com todo o código necessário para que nossa aplicação funcione de forma esperada, ao invés, teremos pequenos .js's separados:

main.jshome.js
a.js
b.js
c.js

Ao grosso modo, teremos algo parecido com o exemplo acima (não que seja assim, foi apenas um exemplo para um melhor entendimento).

Dessa maneira, conseguimos carregar os componentes de forma separada, ou seja, apenas quando forem necessários, por exemplo:

  1. Ao acessar a rota / faz o download do arquivo .js responsável pelo HomeComponent e o renderiza.
  2. Ao acessar a rota /a faz o download do arquivo .js responsável pelo AComponent e o renderiza.
  3. Ao acessar a rota /b faz o download do arquivo .js responsável pelo BComponent e o renderiza.
  4. Ao acessar a rota /c faz o download do arquivo .js responsável pelo CComponent e o renderiza.

Aqui já vemos uso de outra técnica, conhecida como Lazy Loading (carregamento preguiçoso), em outras palavras, é a técnica realizada para carregar os arquivos .js apenas quando forem necessários (o download será feito apenas uma vez).

Implementando o Code Splitting

Chega de teoria e vamos para a prática, hoje na aplicação para exemplo do post temos um componente referente as rotas da aplicação, o Routes:

Basicamente o mesmo está apenas importando os componentes e mapeando suas rotas. Por onde começamos a implementar o code splitting? O primeiro passo, será mudar a forma que estamos importando os componentes, não podemos mais importá-los diretamente através dos modules (módulos).

Precisamos começar a fazer uso da função lazy do React, para isso, devemos importá-la:

Mas, como utilizá-la? Agora nossos componentes devem ser variáveis que irão receber o componente através da função lazy:

O mesmo pode ser feito para todos os componentes dos quais queremos realizar o code splitting e lazy loading:

Mas, ao terminar a mudança e salvar, temos um problema:

Afinal, o que está acontecendo?

Conhecendo o componente Suspense

Bom, vamos entender um pouco melhor nosso problema, basicamente nosso erro é:

Um componente do React foi suspenso enquanto renderizava, porque não adicionamos o componente Suspense como pai de nosso componente que será carregado através de lazy loading.

Para resolver o problema simplesmente podemos encapsular nossas rotas dentro do componente Suspense:

Agora, se tentarmos utilizar a aplicação novamente, temos o seguinte resultado:

E podemos ver que nossos arquivos .js são baixados apenas quando necessários (quando acessarmos a rota de cada componente):

Será que agora tudo está correto? Não, temos apenas um pequeno detalhe para corrigir, ao carregar a aplicação, se olharmos o console do navegador:

Recebemos uma mensagem de erro, o que está acontecendo é que agora estamos passando um Object para a propridade component do componente Route, porém, o mesmo espera que essa props seja uma função (function). Podemos resolver o problema de forma muito simples, apenas encapsulando nossos componentes com uma arrow function:

Agora, podemos utilizar e navegar pela aplicão sem erro nenhum em nosso console.

Mas, afinal, porque tivemos que utilizar o componente Suspense?

Saiba mais

Porque foi necessário o uso do Suspense? Ele foi criado pensando em melhorar a usabilidade de nossa aplicação, geralmente enquanto uma request está sendo feita ou um componente não está pronto para ser renderizado, nós mostramos um Spinner (ou qualquer outro nome que você use) para o usuário. Dessa maneira conseguimos dizer para ele:

Olha, calma, a aplicação está processando…

E quando de fato tudo termina e esta pronto, nosso componente é renderizado com suas informações.

Pensando em facilitar esse tipo de implementação para nós desenvolvedores, foi criado o componente Suspense que através do fallback faz exatamente isso, ele recebe um componente que será mostrado enquanto o download do arquivo .js está sendo feito e a renderização do nosso componente ainda não foi realizada.

Não percebemos muito isso, pois nossa internet é rápida, mas, graças as ferramentas do Google Chrome podemos simular uma internet 3G, assim conseguiremos ver o resultado de forma mais clara:

Para o exemplo do post apenas foi passado uma tag h1 dizendo que está sendo renderizado, mas, em um exemplo do mundo real, o mesmo seria um Spinner (como eu chamo esse tipo de componente).

Conclusão

Nesse post mostrei e expliquei um pouco sobre Code Splitting e Lazy Loading, duas técnicas para melhorar a performance e desempenho da nossa aplicação, pois, os arquivos .js do projeto serão menores e apenas serão baixados quando forem necessários. Com isso, também conseguimos economizar um pouco a internet do usuário (caso seja móvel).

Se você gostou do projeto, pode encontrá-lo em meu github.

Não deixe de comentar ou me procurar em redes sociais para falarmos um pouco sobre o assunto ou tirar alguma dúvida.

Até a próxima.

Publicado originalmente em blog.matheuscastiglioni.com.br em 30 de novembro de 2018.

--

--

Matheus Castiglioni
CollabCode

Apaixonado pelo mundo dos códigos e um eterno estudante, gosto de aprender e saber um pouco de tudo, aquela curiosidade de saber como tudo funciona.