Meningkatkan performa pemuatan halaman Next.js dan Gatsby dengan pemotongan terperinci

Strategi chunking webpack yang lebih baru di Next.js dan Gatsby meminimalkan kode duplikat untuk meningkatkan performa pemuatan halaman.

Chrome berkolaborasi dengan alat dan framework dalam ekosistem open source JavaScript. Sejumlah pengoptimalan yang lebih baru baru-baru ini ditambahkan untuk meningkatkan performa pemuatan Next.js dan Gatsby. Artikel ini membahas strategi pemecahan yang lebih terperinci yang kini dikirimkan secara default di kedua framework.

Pengantar

Seperti banyak framework web, Next.js dan Gatsby menggunakan webpack sebagai bundler inti mereka. webpack v3 memperkenalkan CommonsChunkPlugin untuk memungkinkan modul output yang dibagikan di antara berbagai titik entri dalam satu (atau beberapa) chunk "commons" (atau chunk). Kode bersama dapat didownload secara terpisah dan disimpan di cache browser sejak awal yang dapat menghasilkan performa pemuatan yang lebih baik.

Pola ini menjadi populer dengan banyak framework aplikasi web satu halaman yang mengadopsi konfigurasi entrypoint dan bundle yang terlihat seperti ini:

Konfigurasi titik entri dan paket umum

Meskipun praktis, konsep menggabungkan semua kode modul bersama ke dalam satu chunk memiliki batasannya. Modul yang tidak dibagikan di setiap titik entri dapat didownload untuk rute yang tidak menggunakannya, sehingga lebih banyak kode yang didownload daripada yang diperlukan. Misalnya, saat page1 memuat chunk common, kode untuk moduleC dimuat meskipun page1 tidak menggunakan moduleC. Oleh karena itu, bersama dengan beberapa alasan lainnya, webpack v4 menghapus plugin tersebut dan menggantinya dengan plugin baru: SplitChunksPlugin.

Pengelompokan yang Lebih Baik

Setelan default untuk SplitChunksPlugin berfungsi dengan baik untuk sebagian besar pengguna. Beberapa chunk terpisah dibuat bergantung pada sejumlah kondisi untuk mencegah pengambilan kode duplikat di beberapa rute.

Namun, banyak framework web yang menggunakan plugin ini masih mengikuti pendekatan "single-commons" untuk pemisahan chunk. Misalnya, Next.js akan membuat paket commons yang berisi modul apa pun yang digunakan di lebih dari 50% halaman dan semua dependensi framework (react, react-dom, dan sebagainya).

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)[\\/]/,
      },
    },
  },

Meskipun menyertakan kode yang bergantung pada framework ke dalam chunk bersama berarti kode tersebut dapat didownload dan di-cache untuk titik entri apa pun, heuristik berbasis penggunaan untuk menyertakan modul umum yang digunakan di lebih dari setengah halaman tidak terlalu efektif. Mengubah rasio ini hanya akan menghasilkan salah satu dari dua hasil:

  • Jika Anda mengurangi rasio, lebih banyak kode yang tidak perlu akan didownload.
  • Jika Anda meningkatkan rasio, lebih banyak kode yang diduplikasi di beberapa rute.

Untuk mengatasi masalah ini, Next.js mengadopsi konfigurasi berbeda untukSplitChunksPlugin yang mengurangi kode yang tidak perlu untuk setiap rute.

  • Setiap modul pihak ketiga yang cukup besar (lebih dari 160 KB) akan dibagi menjadi chunk masing-masing
  • Chunk frameworks terpisah dibuat untuk dependensi framework (react, react-dom, dan sebagainya)
  • Chunk bersama dibuat sebanyak yang diperlukan (hingga 25)
  • Ukuran minimum untuk menghasilkan potongan diubah menjadi 20 KB

Strategi pemecahan per bagian yang terperinci ini memberikan manfaat berikut:

  • Waktu pemuatan halaman ditingkatkan. Memancarkan beberapa chunk bersama, bukan satu chunk, meminimalkan jumlah kode yang tidak diperlukan (atau duplikat) untuk setiap titik entri.
  • Peningkatan penyimpanan dalam cache selama navigasi. Memisahkan dependensi framework dan library besar menjadi beberapa bagian terpisah akan mengurangi kemungkinan invalidasi cache karena keduanya cenderung tidak berubah hingga upgrade dilakukan.

Anda dapat melihat seluruh konfigurasi yang diterapkan Next.js di webpack-config.ts.

Lebih banyak permintaan HTTP

SplitChunksPlugin menentukan dasar untuk chunking terperinci, dan menerapkan pendekatan ini ke framework seperti Next.js bukanlah konsep yang sepenuhnya baru. Namun, banyak framework masih terus menggunakan satu strategi heuristik dan paket "umum" karena beberapa alasan. Hal ini termasuk kekhawatiran bahwa banyak permintaan HTTP dapat memengaruhi performa situs secara negatif.

Browser hanya dapat membuka sejumlah koneksi TCP terbatas ke satu origin (6 untuk Chrome), jadi meminimalkan jumlah potongan yang dihasilkan oleh bundler dapat memastikan bahwa jumlah total permintaan tetap di bawah batas ini. Namun, hal ini hanya berlaku untuk HTTP/1.1. Multiplexing di HTTP/2 memungkinkan beberapa permintaan di-streaming secara paralel menggunakan satu koneksi melalui satu origin. Dengan kata lain, kita umumnya tidak perlu khawatir tentang membatasi jumlah potongan yang dikeluarkan oleh bundler.

Semua browser utama mendukung HTTP/2. Tim Chrome dan Next.js ingin melihat apakah peningkatan jumlah permintaan dengan membagi satu paket "commons" Next.js menjadi beberapa chunk bersama akan memengaruhi performa pemuatan dengan cara apa pun. Mereka memulai dengan mengukur performa satu situs sambil mengubah jumlah maksimum permintaan paralel menggunakan properti maxInitialRequests.

Performa pemuatan halaman dengan peningkatan jumlah permintaan

Dalam rata-rata tiga kali menjalankan beberapa uji coba di satu halaman web, waktu load, start-render dan First Contentful Paint tetap sama saat memvariasikan jumlah permintaan awal maksimum (dari 5 hingga 15). Menariknya, kami hanya melihat sedikit overhead performa setelah membagi secara agresif menjadi ratusan permintaan.

Performa pemuatan halaman dengan ratusan permintaan

Hal ini menunjukkan bahwa tetap berada di bawah nilai minimum yang andal (20~25 permintaan) akan mencapai keseimbangan yang tepat antara performa pemuatan dan efisiensi caching. Setelah beberapa pengujian dasar, 25 dipilih sebagai jumlah maxInitialRequest.

Mengubah jumlah maksimum permintaan yang terjadi secara paralel menghasilkan lebih dari satu paket bersama, dan memisahkannya dengan tepat untuk setiap titik entri secara signifikan mengurangi jumlah kode yang tidak diperlukan untuk halaman yang sama.

Pengurangan payload JavaScript dengan peningkatan chunking

Eksperimen ini hanya tentang mengubah jumlah permintaan untuk melihat apakah ada efek negatif pada performa pemuatan halaman. Hasilnya menunjukkan bahwa menyetel maxInitialRequests ke 25 di halaman pengujian adalah yang paling optimal karena mengurangi ukuran payload JavaScript tanpa memperlambat halaman. Jumlah total JavaScript yang diperlukan untuk melakukan hidrasi halaman tetap hampir sama, yang menjelaskan mengapa performa pemuatan halaman tidak selalu meningkat dengan berkurangnya jumlah kode.

webpack menggunakan 30 KB sebagai ukuran minimum default agar potongan dapat dibuat. Namun, menggabungkan nilai maxInitialRequests sebesar 25 dengan ukuran minimum 20 KB justru menghasilkan penyimpanan dalam cache yang lebih baik.

Pengurangan ukuran dengan potongan terperinci

Banyak framework, termasuk Next.js, mengandalkan perutean sisi klien (ditangani oleh JavaScript) untuk menyisipkan tag skrip yang lebih baru untuk setiap transisi rute. Namun, bagaimana cara mereka menentukan potongan dinamis ini pada waktu build?

Next.js menggunakan file manifes build sisi server untuk menentukan chunk output mana yang digunakan oleh berbagai titik entri. Untuk memberikan informasi ini kepada klien juga, file manifes build sisi klien yang disingkat dibuat untuk memetakan semua dependensi untuk setiap titik entri.

// 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}`)) || []
  )
}
Output beberapa chunk bersama dalam aplikasi Next.js.

Strategi chunking terperinci yang lebih baru ini pertama kali diluncurkan di Next.js di balik tanda, yang diuji pada sejumlah pengguna awal. Banyak yang melihat pengurangan signifikan pada total JavaScript yang digunakan untuk seluruh situs mereka:

Situs Total Perubahan JS % Perbedaan
https://www.barnebys.com/ -238 KB -23%
https://sumup.com/ -220 KB -30%
https://www.hashicorp.com/ -11 MB -71%
Pengurangan ukuran JavaScript - di semua rute (dikompresi)

Versi final dikirim secara default di versi 9.2.

Gatsby

Gatsby dulu mengikuti pendekatan yang sama dalam menggunakan heuristik berbasis penggunaan untuk menentukan modul umum:

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)[\\/]/,
      },

Dengan mengoptimalkan konfigurasi webpack untuk menerapkan strategi chunking terperinci yang serupa, mereka juga melihat pengurangan JavaScript yang cukup besar di banyak situs besar:

Situs Total Perubahan JS % Perbedaan
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%
Pengurangan ukuran JavaScript - di semua rute (dikompresi)

Lihat PR untuk memahami cara mereka menerapkan logika ini ke konfigurasi webpack, yang dikirimkan secara default di v2.20.7.

Kesimpulan

Konsep pengiriman potongan terperinci tidak khusus untuk Next.js, Gatsby, atau bahkan webpack. Semua orang harus mempertimbangkan untuk meningkatkan strategi chunking aplikasi mereka jika mengikuti pendekatan paket "umum" yang besar, terlepas dari framework atau bundler modul yang digunakan.

  • Jika Anda ingin melihat pengoptimalan chunking yang sama diterapkan ke aplikasi React vanilla, lihat aplikasi React contoh ini. Aplikasi ini menggunakan versi sederhana dari strategi chunking terperinci dan dapat membantu Anda mulai menerapkan logika yang sama ke situs Anda.
  • Untuk Rollup, chunk dibuat secara terperinci secara default. Lihat manualChunks jika Anda ingin mengonfigurasi perilaku secara manual.