最新のブラウザに最新のコードを提供し、ページの読み込みを高速化する

この Codelab では、ユーザーが無作為に猫を評価できるシンプルなアプリのパフォーマンスを改善します。コードのトランスパイル量を最小限に抑えて JavaScript バンドルを最適化する方法について学びます。

アプリのスクリーンショット

サンプルアプリでは、各猫の好み度を示す単語や絵文字を選択できます。ボタンをクリックすると、現在の猫の画像の下にボタンの値が表示されます。

測定

最適化を追加する前に、ウェブサイトを検査することをおすすめします。

  1. サイトをプレビューするには、[View App] を押してから、[Fullscreen] 全画面表示 を押します。
  2. Ctrl+Shift+J(Mac の場合は Command+Option+J)キーを押して DevTools を開きます。
  3. [ネットワーク] タブをクリックします。
  4. [キャッシュを無効にする] チェックボックスをオンにします。
  5. アプリを再読み込みします。

元のバンドルサイズ リクエスト

このアプリでは 80 KB を超える容量が使用されています。バンドルの一部が使用されていないかどうかを確認します。

  1. Control+Shift+P(Mac では Command+Shift+P)を押して [Command] メニューを開きます。 コマンド メニュー

  2. Show Coverage を入力して Enter キーを押すと、[カバレッジ] タブが表示されます。

  3. [Coverage] タブで [Reload] をクリックして、カバレッジをキャプチャしながらアプリケーションを再読み込みします。

    コードカバレッジを使用してアプリを再読み込みする

  4. 使用されたコードの量と、メインバンドルの読み込み量を比較します。

    バンドルのコード カバレッジ

バンドルの半分以上(44 KB)は使用されていません。これは、古いブラウザでアプリケーションが動作するように、コードの多くがポリフィルで構成されているためです。

@babel/preset-env を使用する

JavaScript 言語の構文は、ECMAScript または ECMA-262 という標準に準拠しています。仕様の新しいバージョンは毎年リリースされ、提案プロセスに合格した新機能が含まれています。主要なブラウザごとに、これらの機能をサポートする段階は異なります。

このアプリケーションでは、次の ES2015 機能が使用されています。

次の ES2017 機能も使用されています。

src/index.js のソースコードを見ると、これらすべての使用例を確認できます。

これらの機能はすべて最新バージョンの Chrome でサポートされていますが、これらをサポートしていない他のブラウザについてはどうなるでしょうか。アプリに含まれている Babel は、新しい構文を含むコードを古いブラウザや環境でも理解できるコードにコンパイルするために使用される最も一般的なライブラリです。この処理は 2 つの方法で行われます。

  • Polyfill は、新しい ES2015 以降の関数をエミュレートするために含まれています。これにより、ブラウザでサポートされていない API でも使用できるようになります。以下に、Array.includes メソッドのポリフィルの例を示します。
  • プラグインは、ES2015 以降のコードを古い ES5 構文に変換するために使用されます。これらは構文に関連する変更(矢印関数など)であるため、ポリフィルでエミュレートすることはできません。

package.json で、どの Babel ライブラリが含まれているかを確認します。

"dependencies": {
  "@babel/polyfill": "^7.0.0"
},
"devDependencies": {
  //...
  "babel-loader": "^8.0.2",
  "@babel/core": "^7.1.0",
  "@babel/preset-env": "^7.1.0",
  //...
}
  • @babel/core はコア Babel コンパイラです。これにより、すべての Babel 構成がプロジェクトのルートの .babelrc で定義されます。
  • babel-loader は、webpack ビルドプロセスに Babel を含めます。

webpack.config.js で、babel-loader がルールとしてどのように含まれているかを確認します。

module: {
  rules: [
    //...
    {
      test: /\.js$/,
      exclude: /node_modules/,
      loader: "babel-loader"
    }
  ]
},
  • @babel/polyfill は、新しい ECMAScript 機能に必要なすべての polyfill を提供するため、それらをサポートしていない環境でも機能します。src/index.js. の一番上にすでにインポートされている
import "./style.css";
import "@babel/polyfill";
  • @babel/preset-env は、ターゲットとして選択されたブラウザまたは環境に必要な変換とポリフィルを特定します。

Babel 構成ファイル .babelrc で、どのように含まれているかを確認します。

{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": "last 2 versions"
      }
    ]
  ]
}

これは Babel と webpack の設定です。webpack とは異なるモジュール バンドラを使用している場合は、Babel をアプリに含める方法をご覧ください。

.babelrctargets 属性は、ターゲットとするブラウザを識別します。@babel/preset-env は browserslist と統合されています。つまり、このフィールドで使用できる互換性のあるクエリの一覧については、browserlist のドキュメントをご覧ください。

"last 2 versions" 値は、すべてのブラウザの過去 2 つのバージョンのアプリケーションのコードをトランスパイルします。

デバッグ

ブラウザのすべての Babel ターゲットと、含まれているすべての変換とポリフィルをすべて確認するには、debug フィールドを .babelrc: に追加します。

{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": "last 2 versions",
        "debug": true
      }
    ]
  ]
}
  • [ツール] をクリックします。
  • [ログ] をクリックします。

アプリケーションを再読み込みし、エディタの下部にある Glitch ステータス ログを確認します。

ターゲットとするブラウザ

Babel は、コードがコンパイルされたすべてのターゲット環境など、コンパイル プロセスに関する多くの詳細情報をコンソールにログに記録します。

ターゲットに設定したブラウザ

Internet Explorer などのサポートが終了したブラウザもこのリストに含まれています。これは問題です。サポートされていないブラウザには新しい機能が追加されず、Babel は引き続き特定の構文をトランスパイルします。ユーザーがこのブラウザを使用してサイトにアクセスしていない場合、バンドルのサイズが不必要に増加します。

Babel はまた、使用された変換プラグインのリストをログに記録します。

使用されているプラグインのリスト

かなり長いリストですが、これらは、Babel が ES2015 以降の構文をターゲットとするすべてのブラウザの古い構文に変換するために使用するすべてのプラグインです。

ただし、使用されている特定のポリフィルは表示されていません。

ポリフィルが追加されていない

これは、@babel/polyfill 全体が直接インポートされるためです。

ポリフィルを個別に読み込む

デフォルトでは、@babel/polyfill をファイルにインポートする際に、完全な ES2015+ 環境に必要なすべてのポリフィルが Babel に含まれています。ターゲット ブラウザに必要な特定のポリフィルをインポートするには、構成に useBuiltIns: 'entry' を追加します。

{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": "last 2 versions",
        "debug": true
        "useBuiltIns": "entry"
      }
    ]
  ]
}

アプリケーションを再読み込みします。含まれる具体的なポリフィルがすべて表示されます。

インポートされたポリフィルのリスト

"last 2 versions" に必要なポリフィルのみが含まれるようになりましたが、それでも非常に長いリストです。これは、新しい機能のすべてについて、ターゲット ブラウザに必要なポリフィルが引き続き含まれているためです。属性の値を usage に変更して、コードで使用されている機能に必要なものだけを含めます。

{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": "last 2 versions",
        "debug": true,
        "useBuiltIns": "entry"
        "useBuiltIns": "usage"
      }
    ]
  ]
}

これにより、必要に応じてポリフィルが自動的に含まれます。つまり、src/index.js.@babel/polyfill インポートを削除できます。

import "./style.css";
import "@babel/polyfill";

これで、アプリケーションに必要なポリフィルだけが含まれるようになりました。

自動的に含まれるポリフィルのリスト

アプリケーション バンドルのサイズが大幅に削減されます。

バンドルサイズを 30.1 KB に縮小

サポートされているブラウザのリストを絞り込む

含まれているブラウザ ターゲットの数は依然としてかなり多く、Internet Explorer などの廃止されたブラウザを使用しているユーザーは多くありません。構成を次のように更新します。

{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": "last 2 versions",
        "targets": [">0.25%", "not ie 11"],
        "debug": true,
        "useBuiltIns": "usage",
      }
    ]
  ]
}

取得したバンドルの詳細を確認します。

バンドルのサイズが 30.0 KB

アプリケーションは非常に小さいため、これらの変更にはさほど違いはありません。ただし、ブラウザのマーケット シェアの割合(">0.25%" など)を使用して、ユーザーが使用していないと思われるブラウザを除外することをおすすめします。詳しくは、ジェームス カイルによる「過去 2 つのバージョン」が有害とみなされるをご覧ください。

<script type="module"> を使用する

まだ改善の余地があります。いくつかの未使用のポリフィルが削除されましたが、一部のブラウザでは不要なポリフィルが数多く出荷されています。モジュールを使用すると、新しい構文を記述し、不要なポリフィルを使用せずにブラウザに直接出荷できます。

JavaScript モジュールは、すべての主要ブラウザでサポートされている比較的新しい機能です。type="module" 属性を使用してモジュールを作成し、他のモジュールからインポートおよびエクスポートするスクリプトを定義できます。例:

// math.mjs
export const add = (x, y) => x + y;

<!-- index.html -->
<script type="module">
  import { add } from './math.mjs';

  add(5, 2); // 7
</script>

新しい ECMAScript 機能の多くは、JavaScript モジュールをサポートする環境ですでにサポートされています(Babel は必要ありません)。つまり、Babel 構成を変更して、アプリケーションの 2 つの異なるバージョンをブラウザに送信することができます。

  • モジュールをサポートする新しいブラウザで動作し、ほとんどトランスパイルされていないがファイルサイズが小さいモジュールを含むバージョン
  • 以前のブラウザで動作する、より大きなトランスパイル済みスクリプトを含むバージョン

Babel で ES モジュールを使用する

2 つのバージョンのアプリに別々の @babel/preset-env 設定を適用するには、.babelrc ファイルを削除します。アプリケーションのバージョンごとに 2 つの異なるコンパイル形式を指定することで、Babel 設定を webpack 構成に追加できます。

まず、以前のスクリプトの構成を webpack.config.js に追加します。

const legacyConfig = {
  entry,
  output: {
    path: path.resolve(__dirname, "public"),
    filename: "[name].bundle.js"
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,
        loader: "babel-loader",
        options: {
          presets: [
            ["@babel/preset-env", {
              useBuiltIns: "usage",
              targets: {
                esmodules: false
              }
            }]
          ]
        }
      },
      cssRule
    ]
  },
  plugins
}

"@babel/preset-env"targets 値を使用する代わりに、値が falseesmodules が使用されています。つまり Babel には、ES モジュールをまだサポートしていないすべてのブラウザをターゲットにするために必要な変換とポリフィルがすべて含まれています。

entrycssRulecorePlugins オブジェクトを webpack.config.js ファイルの先頭に追加します。これらはすべて、ブラウザに提供されるモジュールとレガシー スクリプトの間で共有されます。

const entry = {
  main: "./src"
};

const cssRule = {
  test: /\.css$/,
  use: ExtractTextPlugin.extract({
    fallback: "style-loader",
    use: "css-loader"
  })
};

const plugins = [
  new ExtractTextPlugin({filename: "[name].css", allChunks: true}),
  new HtmlWebpackPlugin({template: "./src/index.html"})
];

同様に、legacyConfig が定義されている次のモジュール スクリプトの構成オブジェクトを作成します。

const moduleConfig = {
  entry,
  output: {
    path: path.resolve(__dirname, "public"),
    filename: "[name].mjs"
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,
        loader: "babel-loader",
        options: {
          presets: [
            ["@babel/preset-env", {
              useBuiltIns: "usage",
              targets: {
                esmodules: true
              }
            }]
          ]
        }
      },
      cssRule
    ]
  },
  plugins
}

主な違いは、出力ファイル名に .mjs ファイル拡張子が使用される点です。ここで esmodules 値が true に設定されているため、このモジュールに出力されるコードは、使用されるすべての機能がモジュールをサポートするブラウザですでにサポートされているため、この例では変換されず、より小さくコンパイルされたスクリプトになります。

ファイルの一番最後に、両方の構成を 1 つの配列にエクスポートします。

module.exports = [
  legacyConfig, moduleConfig
];

これにより、サポートしているブラウザ用の小さいモジュールと、古いブラウザ用の大きなトランスパイルされたスクリプトの両方がビルドされます。

モジュールをサポートしているブラウザでは、nomodule 属性を持つスクリプトは無視されます。逆に、モジュールをサポートしていないブラウザでは、type="module" を含むスクリプト要素は無視されます。つまり、モジュールとコンパイル済みのフォールバックを含めることができます。理想的には、2 つのバージョンのアプリケーションが index.html に次のように存在する必要があります。

<script type="module" src="main.mjs"></script>
<script nomodule src="main.bundle.js"></script>

モジュールをサポートするブラウザは main.mjs を取得して実行し、main.bundle.js. は無視します。モジュールをサポートしないブラウザは、その逆を行います。

通常のスクリプトとは異なり、モジュール スクリプトはデフォルトで常に延期されることに注意してください。 同等の nomodule スクリプトも遅延し、解析後にのみ実行されるようにするには、defer 属性を追加する必要があります。

<script type="module" src="main.mjs"></script>
<script nomodule src="main.bundle.js" defer></script>

最後に、モジュールとレガシー スクリプトにそれぞれ module 属性と nomodule 属性を追加し、webpack.config.js の一番上に ScriptExtHtmlWebpackPlugin をインポートします。

const path = require("path");

const webpack = require("webpack");
const HtmlWebpackPlugin = require("html-webpack-plugin");
const ScriptExtHtmlWebpackPlugin = require("script-ext-html-webpack-plugin");

次に、このプラグインを含めるように、構成の plugins 配列を更新します。

const plugins = [
  new ExtractTextPlugin({filename: "[name].css", allChunks: true}),
  new HtmlWebpackPlugin({template: "./src/index.html"}),
  new ScriptExtHtmlWebpackPlugin({
    module: /\.mjs$/,
    custom: [
      {
        test: /\.js$/,
        attribute: 'nomodule',
        value: ''
    },
    ]
  })
];

これらのプラグイン設定により、すべての .mjs スクリプト要素に type="module" 属性が追加され、すべての .js スクリプト モジュールに nomodule 属性が追加されます。

HTML ドキュメントでモジュールを提供する

最後に、従来のスクリプトと最新のスクリプトの両方の要素を HTML ファイルに出力する必要があります。残念ながら、最終的な HTML ファイルを作成するプラグイン HTMLWebpackPlugin は、現在、モジュール スクリプトと nomodule スクリプトの両方の出力をサポートしていません。この問題を解決するために、BabelMultiTargetPluginHTMLWebpackMultiBuildPlugin などの回避策や個別のプラグインが作成されていますが、このチュートリアルでは、モジュール スクリプト要素を手動で追加する簡単な方法を使用します。

ファイルの末尾の src/index.js に以下を追加します。

    ...
    </form>
    <script type="module" src="main.mjs"></script>
  </body>
</html>

モジュールをサポートするブラウザ(最新バージョンの Chrome など)でアプリケーションを読み込みます。

新しいブラウザではネットワーク経由で 5.2 KB モジュールが取得されました

モジュールのみがフェッチされ、大部分がトランスパイルされていないため、バンドルのサイズが大幅に小さくなります。他のスクリプト要素はブラウザによって完全に無視されます。

古いブラウザでアプリケーションを読み込むと、必要なすべてのポリフィルと変換を含む、トランスパイルされたサイズの大きいスクリプトのみが取得されます。古いバージョンの Chrome(バージョン 38)で行われたすべてのリクエストのスクリーンショットです。

古いブラウザ用に 30 KB のスクリプトを取得

まとめ

これで、@babel/preset-env を使用して、ターゲット ブラウザに必要なポリフィルのみを提供する方法を理解できました。また、JavaScript モジュールで 2 つの異なるトランスパイルされたバージョンのアプリケーションを配布することで、パフォーマンスをさらに向上させる方法も学びました。この 2 つの手法でバンドルサイズを大幅に削減できることを十分に理解して、最適化に取り掛かりましょう。