今日はいつも通りのパッチ適用日でした。コードを変更せずに npm の依存関係にパッチを当ててアップグレードしたところ、突然単体テストの一部が失敗しました。
なんと!
Jest が予期しないトークンに遭遇したため、テストが失敗しました。 Jest はそのままでは ESM のみのパッケージを処理できないため、失敗しました。実際、Jest は CommonJS で書かれています。
しかし、それは何を意味するのでしょうか?そのためには、CommonJS と ESM が存在する理由を理解する必要があります。
Web 開発の初期には、JavaScript は主に jQuery などのライブラリを使用してドキュメント オブジェクト モデル (DOM) を操作するために使用されていました。ただし、Node.js の導入により、サーバーサイド プログラミングにも JavaScript が使用されるようになりました。この変化により、JavaScript コードベースの複雑さとサイズが増大しました。その結果、JavaScript コードを編成および管理するための構造化された方法の必要性が生じました。このニーズを満たすためにモジュール システムが導入され、開発者がコードを管理可能で再利用可能な単位に分割できるようになりました1.
CommonJS は 2009 年に設立され、当初は ServerJS2 という名前でした。これはサーバーサイド JavaScript 用に設計されており、モジュールを定義するための規則を提供します。 Node.js は CommonJS をデフォルトのモジュール システムとして採用し、バックエンド JavaScript 開発者の間で普及しました。 CommonJS はモジュールのインポートに require を使用し、モジュールのエクスポートに module.exports を使用します。 CommonJS のすべての操作は同期的です。つまり、各モジュールは個別にロードされます。
2015 年、ECMAScript は、主にクライアント側の開発を対象とした、ECMAScript Modules (ESM) と呼ばれる新しいモジュール システムを導入しました。 ESM はインポートおよびエクスポート ステートメントを使用し、その操作は非同期であるため、モジュールを並行してロードできます3。当初、ESM はブラウザー向けに設計されていましたが、CommonJS はサーバー向けに設計されました。これはますます JS エコシステムの標準になりました。現在、最新の JavaScript ランタイムは両方のモジュール システムをサポートしています。ブラウザは 2017 年に ESM をネイティブにサポートし始めました。Typescript でさえ ESM 構文を採用しており、ESM 構文を学習するたびに、無意識のうちに ESM も学習することになります。
真実は、ESM のみのパッケージよりも CommonJS (CJS) のみのパッケージの方がはるかに多いということです4.
ただし、明らかな傾向があります。 ESM 専用パッケージまたはデュアル モジュール パッケージの数は増加傾向にありますが、CJS 専用パッケージの作成数は減少しています。この傾向は、ESM に対する優先度が高まっていることを強調しており、CJS のみのパッケージのうちどれだけが積極的に保守されているのかという疑問を引き起こしています。
CommonJS と ESM の興味深い比較には、パフォーマンス ベンチマークが含まれます。 CommonJS は同期的な性質があるため、require ステートメントと import ステートメントを直接使用すると高速になります。次の例を考えてみましょう:
// CommonJS -> s3-get-files.cjs const s3 = require('@aws-sdk/client-s3'); new s3.S3Client({ region: 'eu-central-1' }); // ESM -> s3-get-files.mjs import { S3Client } from '@aws-sdk/client-s3'; new S3Client({ region: 'eu-central-1' });
デュアルモジュールをサポートしているため、aws-sdk S3-Client を使用しました。ここではクライアントをインスタンス化し、node:
で実行します。
hyperfine --warmup 10 --style color 'node s3-get-files.cjs' 'node s3-get-files.mjs' Benchmark 1: node s3-get-files.cjs Time (mean ± σ): 82.6 ms ± 3.7 ms [User: 78.5 ms, System: 16.7 ms] Range (min … max): 78.0 ms … 93.6 ms 37 runs Benchmark 2: node s3-get-files.mjs Time (mean ± σ): 93.9 ms ± 4.0 ms [User: 98.3 ms, System: 18.1 ms] Range (min … max): 88.1 ms … 104.8 ms 32 runs Summary node s3-get-files.cjs ran 1.14 ± 0.07 times faster than node s3-get-files.mjs
ご覧のとおり、s3-get-files.cjs と CommonJS の実行が高速化されています。
Buns のブログ投稿からインスピレーションを得ました。
ただし、JS ライブラリを製品化する場合は、それをバンドルする必要があります。それ以外の場合は、すべてのnode_modulesを出荷することになります。 CJSとESMにバンドルできるためesbuildを使用します。 次に、バンドルされたバージョンで同じベンチマークを実行してみましょう。
hyperfine --warmup 10 --style color 'node s3-bundle.cjs' 'node s3-bundle.mjs' Benchmark 1: node s3-bundle.cjs Time (mean ± σ): 62.1 ms ± 2.5 ms [User: 53.8 ms, System: 6.7 ms] Range (min … max): 59.5 ms … 74.5 ms 45 runs Warning: Statistical outliers were detected. Consider re-running this benchmark on a quiet system without any interferences from other programs. It might help to use the '--warmup' or '--prepare' options. Benchmark 2: node s3-bundle.mjs Time (mean ± σ): 45.3 ms ± 2.2 ms [User: 38.1 ms, System: 5.6 ms] Range (min … max): 43.0 ms … 59.2 ms 62 runs Warning: Statistical outliers were detected. Consider re-running this benchmark on a quiet system without any interferences from other programs. It might help to use the '--warmup' or '--prepare' options. Summary node s3-bundle.mjs ran 1.37 ± 0.09 times faster than node s3-bundle.cjs
ご覧のとおり、s3-bundle.mjs は s3-bundle.cjs よりも高速になりました。 ESM ファイルは、バンドルされていない CommonJS ファイルよりもさらに高速になりました。これは、効率的なツリーシェーキング (未使用のコードを削除するプロセス) によってファイル サイズが小さくなり、ロード時間が短縮されるためです。
JavaScript モジュールの将来は間違いなく ESM に傾いています。これは、新しい NodeJS プロジェクト、または React プロジェクトを作成するときに始まります。すべてのチュートリアルと記事では import ステートメントが使用されており、これが ESM です。多くの CommonJS パッケージが既存しているにもかかわらず、パフォーマンス上の利点と最新の構文を求めて ESM を採用する開発者や保守者が増えるにつれて、傾向は変わりつつあります。もう 1 つの疑問は、これらの CJS のみのプロジェクトがどれくらいまだ維持されているかということです。
ESM は、NodeJS、Bun、Deno などのあらゆるランタイムで、サーバー上で実行せずにブラウザーで動作する標準です。ブラウザは ESM を理解するため、Babel 経由で CommonJS に変換する必要はありません。 Babel を使用して別の ECMAScript バージョンに変換することはできますが、CJS に変換しないでください。
現在のすべてのランタイムと 2017 年以降のブラウザは ESM を理解するため、ESM のみで開発する必要があります。
コードが壊れた場合、従来の問題が発生している可能性があります。別のツールまたはパッケージの使用を検討してください。たとえば、Jest から vitest に移行したり、ExpressJS から h3 に移行したりできます。構文は同じままです。唯一の違いは import ステートメントです。
重要なポイント:
まず、この要点に従うか、ここでインスピレーションを与える学習を得ることができます。
JavaScript のより良い未来のために、ESM を取り入れましょう!
プレゼンテーション
https://www.freecodecamp.org/news/javascript-es-modules-and-module-bundlers/#why-use-modules ↩
https://deno.com/blog/commonjs-is-hurting-javascript ↩
https://tc39.es/ecma262/#sec-overview ↩
https://twitter.com/wooorm/status/1759918205928194443 ↩
免責事項: 提供されるすべてのリソースの一部はインターネットからのものです。お客様の著作権またはその他の権利および利益の侵害がある場合は、詳細な理由を説明し、著作権または権利および利益の証拠を提出して、電子メール [email protected] に送信してください。 できるだけ早く対応させていただきます。
Copyright© 2022 湘ICP备2022001581号-3