Uma estratégia de divisão do webpack mais recente no Next.js e no Gatsby minimiza o código duplicado para melhorar o desempenho de carregamento da página.
O Chrome colabora com ferramentas e frameworks no ecossistema de código aberto do JavaScript. Várias otimizações mais recentes foram adicionadas recentemente para melhorar o desempenho de carregamento do Next.js e do Gatsby. Este artigo aborda uma estratégia de divisão granular aprimorada que agora é enviada por padrão nos dois frameworks.
Introdução
Como muitos frameworks da Web, o Next.js e o Gatsby usam o webpack como o principal
bundler. O webpack v3 introduziu o
CommonsChunkPlugin
para permitir a
saída de módulos compartilhados entre diferentes pontos de entrada em um único (ou alguns) bloco "comuns"
(ou blocos). O código compartilhado pode ser transferido por download separadamente e armazenado no cache do navegador com antecedência, o que pode
resultar em um melhor desempenho de carregamento.
Esse padrão se tornou popular com muitos frameworks de aplicativos de página única que adotam uma configuração de ponto de entrada e de pacotes semelhante a esta:
Embora prático, o conceito de agrupar todo o código de módulo compartilhado em um único bloco tem
limitações. Módulos não compartilhados em todos os pontos de entrada podem ser transferidos por download para rotas que não os usam,
resultando em mais código do que o necessário. Por exemplo, quando page1
carrega
o bloco common
, ele carrega o código para moduleC
, mesmo que page1
não use moduleC
.
Por esse motivo, entre outros, o webpack v4 removeu o plug-in em favor de um novo: SplitChunksPlugin
.
Chunking aprimorado
As configurações padrão de SplitChunksPlugin
funcionam bem para a maioria dos usuários. Vários blocos divididos são
criados dependendo de várias condições
para evitar a busca de código duplicado em várias rotas.
No entanto, muitos frameworks da Web que usam esse plug-in ainda seguem uma abordagem de "comuns" única para a divisão
de blocos. O Next.js, por exemplo, gera um pacote commons
que contém qualquer módulo
usado em mais de 50% das páginas e todas as dependências do framework (react
, react-dom
e assim por diante).
const splitChunksConfigs = {
…
prod: {
chunks: 'all',
cacheGroups: {
default: false,
vendors: false,
commons: {
name: 'commons',
chunks: 'all',
minChunks: totalPages > 2 ? totalPages * 0.5 : 2,
},
react: {
name: 'commons',
chunks: 'all',
test: /[\\/]node_modules[\\/](react|react-dom|scheduler|use-subscription)[\\/]/,
},
},
},
Embora a inclusão de código dependente do framework em um bloco compartilhado signifique que ele pode ser baixado e armazenado em cache para qualquer ponto de entrada, a heurística baseada no uso de incluir módulos comuns usados em mais de metade das páginas não é muito eficaz. Modificar essa proporção só resultaria em um destes dois resultados:
- Se você reduzir a proporção, mais código desnecessário será transferido.
- Se você aumentar a proporção, mais código será duplicado em várias rotas.
Para resolver esse problema, o Next.js adotou uma configuração
diferente para SplitChunksPlugin
, que reduz
o código desnecessário para qualquer rota.
- Qualquer módulo de terceiros suficientemente grande (maior que 160 KB) é dividido em um bloco individual.
- Um bloco
frameworks
separado é criado para dependências de framework (react
,react-dom
e assim por diante). - São criados quantos blocos compartilhados forem necessários (até 25).
- O tamanho mínimo de um fragmento a ser gerado foi alterado para 20 KB
Essa estratégia de divisão em partes menores oferece os seguintes benefícios:
- O tempo de carregamento da página é melhorado. Emitir vários blocos compartilhados, em vez de um único, minimiza a quantidade de código desnecessário (ou duplicado) para qualquer ponto de entrada.
- Melhoria no armazenamento em cache durante as navegações. A divisão de bibliotecas grandes e dependências de framework em partes separadas reduz a possibilidade de invalidação do cache, já que é improvável que ambos mudem até que um upgrade seja feito.
É possível conferir toda a configuração adotada pelo Next.js em webpack-config.ts
.
Mais solicitações HTTP
SplitChunksPlugin
definiu a base para o agrupamento granular, e aplicar essa abordagem a um
framework como o Next.js não era um conceito totalmente novo. No entanto, muitos frameworks ainda continuavam
usando uma única heurística e estratégia de pacote "commons" por alguns motivos. Isso inclui a preocupação de que
muitas outras solicitações HTTP possam afetar negativamente o desempenho do site.
Os navegadores só podem abrir um número limitado de conexões TCP para uma única origem (6 para o Chrome). Portanto, minimizar o número de blocos gerados por um bundler pode garantir que o número total de solicitações permaneça abaixo desse limite. No entanto, isso só é válido para HTTP/1.1. A multiplexação no HTTP/2 permite que várias solicitações sejam transmitidas em paralelo usando uma única conexão em uma única origem. Em outras palavras, geralmente não precisamos nos preocupar em limitar o número de blocos emitidos pelo nosso bundler.
Todos os principais navegadores oferecem suporte a HTTP/2. As equipes do Chrome e do Next.js
queriam saber se aumentar o número de solicitações dividindo o pacote "commons" único do Next.js
em vários blocos compartilhados afetaria a performance de carregamento de alguma forma. Eles começaram medindo a
performance de um único site enquanto modificavam o número máximo de solicitações paralelas usando a
propriedade
maxInitialRequests
.
Em uma média de três execuções de vários testes em uma única página da Web, os tempos de
load
,
início da renderização
e First Contentful Paint permaneceram praticamente os mesmos ao variar a contagem de solicitações iniciais
máximas (de 5 a 15). Curiosamente, notamos um pequeno overhead de desempenho apenas
após dividir agressivamente para centenas de solicitações.
Isso mostrou que permanecer abaixo de um limite confiável (20 a 25 solicitações) atingiu o equilíbrio certo
entre a performance de carregamento e a eficiência do armazenamento em cache. Após alguns testes de referência, 25 foi selecionado como
a contagem de maxInitialRequest
.
Modificar o número máximo de solicitações que acontecem em paralelo resultou em mais de um pacote compartilhado, e separá-los adequadamente para cada ponto de entrada reduziu significativamente a quantidade de código desnecessário para a mesma página.
O objetivo do experimento era modificar o número de solicitações para saber se haveria algum
efeito negativo no desempenho de carregamento da página. Os resultados sugerem que definir maxInitialRequests
como
25
na página de teste foi ideal porque reduziu o tamanho do payload do JavaScript sem diminuir
a velocidade da página. A quantidade total de JavaScript necessária para hidratar a página permaneceu
aproximadamente a mesma, o que explica por que o desempenho de carregamento da página não melhorou necessariamente com a redução
da quantidade de código.
O webpack usa 30 KB como tamanho mínimo padrão para a geração de um fragmento. No entanto, o acoplamento de um
valor maxInitialRequests
de 25 com um tamanho mínimo de 20 KB resultou em um armazenamento em cache melhor.
Redução de tamanho com blocos granulares
Muitos frameworks, incluindo o Next.js, dependem do roteamento do lado do cliente (gerenciado pelo JavaScript) para injetar tags de script mais recentes em cada transição de rota. Mas como eles predeterminam esses blocos dinâmicos no momento da criação?
O Next.js usa um arquivo de manifesto de build do servidor para determinar quais blocos de saída são usados por diferentes pontos de entrada. Para fornecer essas informações ao cliente, um arquivo de manifesto de build resumido do lado do cliente foi criado para mapear todas as dependências de cada ponto de entrada.
// Returns a promise for the dependencies for a particular route
getDependencies (route) {
return this.promisedBuildManifest.then(
man => (man[route] && man[route].map(url => `/_next/${url}`)) || []
)
}
Essa nova estratégia de divisão granular foi lançada pela primeira vez no Next.js com uma flag, onde foi testada em vários usuários iniciais. Muitos tiveram reduções significativas no total de JavaScript usado em todo o site:
Site | Mudança total do JS | Diferença % |
---|---|---|
https://www.barnebys.com/ | -238 KB | -23% |
https://sumup.com/ | -220 KB | -30% |
https://www.hashicorp.com/ | -11 MB | -71% |
A versão final foi enviada por padrão na versão 9.2.
Gatsby
O Gatsby costumava seguir a mesma abordagem de usar uma heurística baseada no uso para definir módulos comuns:
config.optimization = {
…
splitChunks: {
name: false,
chunks: `all`,
cacheGroups: {
default: false,
vendors: false,
commons: {
name: `commons`,
chunks: `all`,
// if a chunk is used more than half the components count,
// we can assume it's pretty global
minChunks: componentsCount > 2 ? componentsCount * 0.5 : 2,
},
react: {
name: `commons`,
chunks: `all`,
test: /[\\/]node_modules[\\/](react|react-dom|scheduler)[\\/]/,
},
Ao otimizar a configuração do webpack para adotar uma estratégia de fragmentação granular semelhante, eles também notaram reduções significativas de JavaScript em muitos sites grandes:
Site | Mudança total do JS | Diferença % |
---|---|---|
https://www.gatsbyjs.org/ | -680 KB | -22% |
https://www.thirdandgrove.com/ | -390 KB | -25% |
https://ghost.org/ | -1,1 MB | -35% |
https://reactjs.org/ | -80 Kb | -8% |
Confira o PR para entender como essa lógica foi implementada na configuração do webpack, que é enviada por padrão na v2.20.7.
Conclusão
O conceito de envio de blocos granulares não é específico do Next.js, do Gatsby ou mesmo do webpack. Todos precisam considerar melhorar a estratégia de divisão do aplicativo se ele seguir uma abordagem de pacote "commons" grande, independentemente do framework ou do agrupador de módulos usado.
- Se você quiser conferir as mesmas otimizações de fragmentação aplicadas a um aplicativo React vanilla, confira este exemplo de app React. Ele usa uma versão simplificada da estratégia de fragmentação granular e pode ajudar você a começar a aplicar o mesmo tipo de lógica ao seu site.
- Para o agrupamento, os blocos são criados de forma granular por padrão. Consulte
manualChunks
se quiser configurar o comportamento manualmente.