JavaScript 起動パフォーマンスのボトルネック分析と解決策
Web 開発では、需要が増加し、コードベースが拡大するにつれて、最終的に公開する Web ページも徐々に拡大します。ただし、この拡張は、より多くの伝送帯域幅を占有することを意味するだけでなく、ユーザーが Web を閲覧するときにパフォーマンスが低下する可能性があることも意味します。ブラウザーは、特定のページが依存するスクリプトをダウンロードした後も、構文分析、解釈、実行の次の手順を実行する必要があります。この記事では、ブラウザの JavaScript 処理を詳細に分析し、アプリケーションの起動時間に影響を与える原因を特定し、私の個人的な経験に基づいて対応する解決策を提案します。振り返ってみると、JavaScript の解析/コンパイル手順を最適化する方法については特に考えていませんでした。パーサーが <script>
タグを検出した後、即座に解析操作を完了すると予想していましたが、これは明らかに夢物語です。次の図は、V8 エンジンの動作原理の概要です:
主要な手順を詳しく分析してみましょう。
アプリケーションの起動時間を遅くしているのは何ですか?
起動フェーズでは、JavaScript エンジンの実行時間のほとんどが構文分析、コンパイル、スクリプトの実行に費やされます。言い換えれば、これらのプロセスによって生じる遅延は、ユーザーの操作の遅延を正確に反映します。たとえば、ユーザーがボタンを見ても、実際にクリックできるようになるまでに数秒かかり、ユーザー エクスペリエンスに大きな影響を与えます。
上の写真は、Chrome Canary に組み込まれた V8 ランタイム呼び出し統計を使用した特定の Web サイトの分析結果です。デスクトップ ブラウザーでは構文解析とコンパイルに依然として時間がかかり、モバイルでは構文解析とコンパイルに時間がかかることに注意してください。ターミナルの方が時間がかかります。実際、Facebook、Wikipedia、Reddit などの大規模な Web サイトで構文解析とコンパイルに費やされる時間は無視できません:
上の図のピンク色の領域は V8 と Blink の C++ で費やされる時間を表し、オレンジと黄色は、それぞれ構文解析とコンパイルの時間の割合を示します。 Facebook の Sebastian Markbage と Google の Rob Wormald も、JavaScript の長い構文解析時間が無視できない問題になっていると Twitter に投稿しました。後者も、これが Angular を開始する際の主な消費の 1 つであると述べています。
モバイル端末の急増に伴い、私たちは残酷な事実に直面しなければなりません。モバイル端末での同じパッケージ本体の解析とコンパイルのプロセスには、デスクトップブラウザの 2 ~ 5 倍の時間がかかります。もちろん、iPhone や Pixel などのハイエンドの携帯電話は、Moto G4 のようなミッドレンジの携帯電話よりもはるかに優れたパフォーマンスを発揮します。これは、テストするときに、周囲のハイエンドの携帯電話だけを使用するのではなく、両方のミッドレンジの携帯電話を考慮する必要があることを思い出させます。 -範囲およびローエンドの携帯電話:
上の図は、一部のデスクトップ ブラウザとモバイル ブラウザの間で 1 MB の JavaScript パッケージ本体の解析時間を比較したものです。さまざまな構成。アプリケーション パッケージ本体がすでに非常に大きい場合、コード分割、TreeShaking、Service Workder キャッシュなどの最新のパッケージ化手法を使用すると、起動時間に大きな影響を与えます。別の観点から見ると、たとえそれが小さなモジュールであっても、コードの書き方が不十分であったり、依存関係ライブラリの使用が不十分であったりすると、メインスレッドはコンパイルや冗長な関数呼び出しに多くの時間を費やすことになります。実際のパフォーマンスのボトルネックを掘り出すには、総合的な評価の重要性を明確に理解する必要があります。
JavaScript 構文の解析とコンパイルは、ほとんどの Web サイトにとってボトルネックになっていますか?
私は Facebook ではありません、という言葉を何度も聞きました。あなたが言及した JavaScript の構文解析とコンパイルは、他の Web サイトにどのような影響を及ぼしますか?私もこの問題に興味があったので、2 か月かけて 6,000 を超える Web サイトを分析しました。これらの Web サイトには、React、Angular、Ember、Vue などの人気のあるフレームワークやライブラリが含まれていました。ほとんどのテストは WebPageTest に基づいているため、これらのテスト結果を簡単に再現できます。
光ファイバーアクセスを備えたデスクトップブラウザではユーザー操作が可能になるまでに約 8 秒かかりますが、3G 環境の Moto G4 ではユーザー操作が可能になるまでに約 16 秒かかります。
ほとんどのアプリケーションは、JavaScript の起動フェーズ (文法解析、コンパイル、実行) に約 4 秒かかります デスクトップ ブラウザでは:
disabled-by-default-v8.runtime_stats
を使用して詳細を理解できます。 V8 の消費時間。 V8 では、この機能の使用方法に関する詳細なガイダンスも提供されます。 disabled-by-default-v8.runtime_stats
来深度了解 V8 的时间消耗情况。V8 也提供了详细的指南来介绍如何使用这个功能。
WebPageTest
WebPageTest 中 Processing Breakdown 页面在我们启用 Chrome > Capture Dev Tools Timeline 时会自动记录 V8 编译、EvaluateScript 以及 FunctionCall 的时间。我们同样可以通过指明disabled-by-default-v8.runtime_stats
WebPageTest

disabled-by-default-v8.runtime_stats
を指定して、ランタイム呼び出し統計を有効にすることもできます。 🎜🎜🎜🎜🎜詳しい手順については、私の要点を参照してください。 🎜ユーザー タイミング
Nolan Lawson が推奨する User Timing API を使用して、文法解析の時間を評価することもできます。ただし、このメソッドは V8 の事前解析プロセスの影響を受ける可能性があります。optimize-js の評価で Nolan のメソッドから学び、スクリプトの最後にランダムな文字列を追加することで、この問題を解決できます。実際のユーザーやデバイスがウェブサイトにアクセスしたときの解析時間を評価するために、Google Analytics に基づく同様の方法を使用しています:
DeviceTiming
Etsy の DeviceTiming ツールは、特定の制限された環境をシミュレートして、ページの構文解析と実行時間を評価できます。 。ローカル スクリプトをインストルメンテーション ツール コードでラップして、ページがさまざまなデバイスからのアクセスをシミュレートできるようにします。詳しい使用法については、Daniel Espeset の記事「モバイル デバイスでの JS 解析と実行のベンチマーク」を参照してください。
JavaScript の解析時間を短縮するにはどうすればよいでしょうか?
JavaScript パッケージ本体のサイズを削減します。また、パッケージ本体が小さいほど解析ワークロードが少なくなるため、解析およびコンパイルのフェーズでのブラウザの時間消費も削減できることも上で説明しました。
コード分割ツールを使用してオンデマンドでコードを渡し、残りのモジュールを遅延ロードします。 PRPL のようなモデルはルートベースのグループ化を促進し、現在 Flipkart、Housing.com、Twitter で広く使用されているため、これがおそらく最良のアプローチです。
スクリプト ストリーミング: 以前、V8 は開発者に
async/defer
を使用して、スクリプト ストリーミングに基づいて 10 ~ 20% のパフォーマンス向上を達成することを推奨していました。この技術により、HTML パーサーは対応するスクリプト読み込みタスクを専用のスクリプト ストリーミング スレッドに割り当てることができるため、ドキュメント解析のブロックが回避されます。 V8 では、ストリーマ スレッドが 1 つしかないため、できるだけ早く大きなモジュールをロードすることをお勧めします。依存関係の解析コストを評価します。 React の代わりに Preact または Inferno を使用するなど、機能は同じで読み込みが速い依存関係を選択するよう最善を尽くす必要があります。これらは、React よりもサイズが小さく、必要な構文解析とコンパイル時間が短くなります。 Paul Lewis も最近の記事でフレームワークの起動コストについて議論しており、これは Sebastian Markbage の声明と一致しています。フレームワークの起動コストを評価する最良の方法は、最初にインターフェイスをレンダリングし、次に削除し、最後に再レンダリングすることです。最初のレンダリング プロセスには分析とコンパイルが含まれます。比較を通じて、フレームワークの起動時の消費量を確認できます。
JavaScript フレームワークが AOT (ahead-of-time) コンパイル モードをサポートしている場合、解析とコンパイルの時間を効果的に短縮できます。 Angular アプリケーションはこのパターンの恩恵を受けます:
最新のブラウザーはどのようにして解析とコンパイルの速度を向上させますか?
落胆しないでください。起動時間を改善する方法に苦労しているのはあなただけではありません。私たちの V8 チームも懸命に取り組んでいます。以前の評価ツールである Octane は、実際のシナリオの優れたシミュレーションであることがわかりました。マイクロ フレームワークとコールド スタートの点で、実際のユーザーの習慣と非常に一致しています。これらのツールに基づいて、V8 チームは過去の作業で起動パフォーマンスの約 25% の向上も達成しました:
このセクションでは、過去数年間に構文解析とコンパイル時間を改善するために使用したテクニックをレビューします。 。 精巧な。
コード キャッシュ
Chrome 42 では、いわゆるコード キャッシュの概念が導入され始めました。これにより、ユーザーがアクセスしたときにスクリプトのクロールを回避できるように、コンパイルされたコードのコピーを保存するメカニズムが提供されます。もう一度ページを開き、これらの手順を解析してコンパイルします。さらに、このメカニズムにより、繰り返しアクセス中のコンパイル時間の約 40% も回避できることもわかりました。ここでいくつかの内容を詳しく紹介します。
コード キャッシュは、72 時間以内に繰り返し実行されるコードをキャッシュします。スクリプトは機能します。
Service Worker のスクリプトの場合、コード キャッシュは 72 時間以内のスクリプトに対しても機能します。
Service Worker を使用してキャッシュ ストレージにキャッシュされたスクリプトの場合、スクリプトが最初に実行されるときにコード キャッシュが有効になることがあります。
つまり、アクティブにキャッシュされた JavaScript コードの場合、最大でも 3 回目の呼び出しで構文分析とコンパイルの手順をスキップできます。 chrome://flags/#v8-cache-strategies-for-cache-storage
を通じて違いを確認するか、js-flags=profile-deserialization
を設定して Chrome を実行しますコードがコード キャッシュからロードされているかどうかを確認します。ただし、コード キャッシュ メカニズムはコンパイル済みコードのみをキャッシュすることに注意してください。これは主に、グローバル変数の設定によく使用されるトップレベル コードを指します。関数定義などの遅延コンパイルされたコードはキャッシュされませんが、IIFE は V8 にも含まれているため、これらの関数もキャッシュできます。 chrome://flags/#v8-cache-strategies-for-cache-storage
来查看其中的差异,也可以设置 js-flags=profile-deserialization
运行 Chrome 来查看代码是否加载自代码缓存。不过需要注意的是,代码缓存机制仅会缓存那些经过编译的代码,主要是指那些顶层的往往用于设置全局变量的代码。而对于类似于函数定义这样懒编译的代码并不会被缓存,不过 IIFE 同样被包含在了 V8 中,因此这些函数也是可以被缓存的。
Script Streaming
Script Streaming允许在后台线程中对异步脚本执行解析操作,可以对于页面加载时间有大概 10% 的提升。上文也提到过,这个机制同样会对同步脚本起作用。
这个特性倒是第一次提及,因此 V8 会允许所有的脚本,即使阻塞型的<script src=''>
脚本也可以由后台线程进行解析。不过缺陷就是目前仅有一个 streaming 后台线程存在,因此我们建议首先解析大的、关键性的脚本。在实践中,我们建议将<script defer>
添加到<head>
块内,这样浏览器引擎就能够尽早地发现需要解析的脚本,然后将其分配给后台线程进行处理。我们也可以查看 DevTools Timeline 来确定脚本是否被后台解析,特别是当你存在某个关键性脚本需要解析的时候,更需要确定该脚本是由 streaming 线程解析的。
语法解析 & 编译优化
我们同样致力于打造更轻量级、更快的解析器,目前 V8 主线程中最大的瓶颈在于所谓的非线性解析消耗。譬如我们有如下的代码片:
(function (global, module) { … })(this, function module() { my functions })
V8 并不知道我们编译主脚本的时候是否需要module
这个模块,因此我们会暂时放弃编译它。而当我们打算编译module
时,我们需要重分析所有的内部函数。这也就是所谓的 V8 解析时间非线性的原因,任何一个处于 N 层深度的函数都有可能被重新分析 N 次。V8 已经能够在首次编译的时候搜集所有内部函数的信息,因此在未来的编译过程中 V8 会忽略所有的内部函数。对于上面这种module

この機能は初めて言及されたため、V8 では、
<script src= をブロックする場合も含め、すべてのスクリプトが許可されます。 ''>
スクリプトはバックグラウンド スレッドによって解析することもできます。ただし、現時点ではストリーミング バックグラウンド スレッドが 1 つしかないという欠点があるため、最初に大規模で重要なスクリプトを解析することをお勧めします。実際には、ブラウザ エンジンが解析する必要があるスクリプトをできるだけ早く検出できるように、<script defer>
を <head>
ブロック内に追加することをお勧めします。次に、それを処理のためにバックグラウンド スレッドに割り当てます。また、DevTools タイムラインをチェックして、スクリプトがバックグラウンドで解析されているかどうかを確認することもできます。特に、解析する必要がある重要なスクリプトがある場合は、そのスクリプトがストリーミング スレッドによって解析されていることを確認する必要があります。 A 
rrreee
V8 メインスクリプトをコンパイルするときにmodule
モジュールが必要かどうかわからないため、一時的にコンパイルを諦めます。 module
をコンパイルする場合は、すべての内部関数を再分析する必要があります。これが、V8 解析時間のいわゆる非線形性の理由です。N レベルの深さの関数は N 回再解析される可能性があります。 V8 は、最初のコンパイル時にすべての内部関数に関する情報を収集できるため、今後のコンパイルではすべての内部関数を無視します。 module
形式の上記の関数については、パフォーマンスが大幅に向上します。詳細については、「V8 パーサー — 設計、課題、および JavaScript の解析の向上」を参照することをお勧めします。 V8 では、起動時に JavaScript のコンパイル プロセスをバックグラウンド スレッドで実行できるようにする適切なオフロード メカニズムも探しています。 プリコンパイル済みの JavaScript?
プリコンパイルは、ブラウザーの実行に必要な最小限の関数セットのみを処理することで起動時間を短縮できますが、このメカニズムは IIFE に直面すると実際には効率を低下させます。エンジンはこれらの関数の前処理を回避したいと考えていますが、optimize-js のようなライブラリよりも効果がはるかに低くなります。 optimize-js はエンジンの前にスクリプトを処理し、すぐに実行される関数に括弧を挿入して実行を高速化します。この種の前処理は、Browserify と Webpack で生成されたパッケージ本体に非常に優れた最適化効果をもたらします。パッケージ本体には、すぐに実行できる多数の小さなモジュールが含まれています。この小さなトリックは V8 が使用したいものではありませんが、対応する最適化メカニズムを現段階で導入する必要があります。
概要
起動フェーズのパフォーマンスは非常に重要であり、解析、コンパイル、実行時間が遅いと、Web ページのパフォーマンスのボトルネックになる可能性があります。この段階でページが費やす時間を評価し、最適化するための適切な方法を選択する必要があります。今後もV8の始動性能向上に全力で取り組んでまいります!
上記は JavaScript 起動時のパフォーマンスのボトルネック分析と解決策の内容です。さらに関連する内容については、PHP 中国語 Web サイト (www.php.cn) に注目してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック









Win11 は Microsoft が発売した最新のオペレーティング システムで、以前のバージョンと比較してインターフェイスのデザインとユーザー エクスペリエンスが大幅に向上しています。ただし、一部のユーザーは、Win11 をインストールした後、中国語の言語パックをインストールできないという問題が発生し、システムで中国語を使用する際に問題が発生したと報告しました。この記事では、ユーザーが中国語をスムーズに使用できるように、Win11 で中国語言語パックをインストールできない問題の解決策をいくつか紹介します。まず、中国語言語パックをインストールできない理由を理解する必要があります。一般的に言えば、Win11

scipy ライブラリのインストールが失敗する理由と解決策、具体的なコード例が必要 Python で科学計算を実行する場合、scipy は非常に一般的に使用されるライブラリであり、数値計算、最適化、統計、および信号処理のための多くの機能を提供します。ただし、scipy ライブラリをインストールするときに、いくつかの問題が発生し、インストールが失敗することがあります。この記事では、scipy ライブラリのインストールが失敗する主な理由を調査し、対応する解決策を提供します。依存パッケージのインストールに失敗しました。scipy ライブラリは、nu などの他の Python ライブラリに依存しています。

OracleNVL 機能の一般的な問題と解決策 Oracle データベースは広く使用されているリレーショナル データベース システムであり、データ処理中に NULL 値を処理する必要があることがよくあります。 NULL 値によって引き起こされる問題に対処するために、Oracle は NULL 値を処理する NVL 関数を提供します。この記事では、NVL 関数の一般的な問題と解決策を紹介し、具体的なコード例を示します。質問 1: NVL 関数の不適切な使用法 NVL 関数の基本構文は、NVL(expr1,default_value) です。

タイトル: Oracle のキャラクタセット変更による文字化け問題を解決する有効な解決策 Oracle データベースでは、キャラクタセットを変更すると、データ内に互換性のない文字が存在することにより文字化けの問題が発生することがよくあります。この問題を解決するには、いくつかの効果的な解決策を採用する必要があります。この記事では、Oracleの文字セット変更による文字化けの問題を解決する具体的な解決策とコード例を紹介します。 1. データをエクスポートして文字セットをリセットする まず、expdp コマンドを使用してデータベース内のデータをエクスポートします。

C++ の機械学習アルゴリズムが直面する一般的な課題には、メモリ管理、マルチスレッド、パフォーマンスの最適化、保守性などがあります。解決策には、スマート ポインター、最新のスレッド ライブラリ、SIMD 命令、サードパーティ ライブラリの使用、コーディング スタイル ガイドラインの遵守、自動化ツールの使用が含まれます。実践的な事例では、Eigen ライブラリを使用して線形回帰アルゴリズムを実装し、メモリを効果的に管理し、高性能の行列演算を使用する方法を示します。

PyCharm は、開発者に広く愛されている強力な Python 統合開発環境です。ただし、PyCharm を使用するときにキーの無効化の問題が発生し、ソフトウェアを正常に使用できなくなる場合があります。この記事では、PyCharm キーの障害に対する解決策を明らかにし、読者がこの問題をすぐに解決できるように具体的なコード例を示します。問題の解決を始める前に、まずキーが無効である理由を理解する必要があります。 PyCharm キーの失敗は通常、ネットワークの問題またはソフトウェア自体が原因です

0xc000007b 正常に起動できない問題を解決する方法 コンピューターを使用しているときに、さまざまなエラー コードに遭遇することがありますが、最も一般的なものの 1 つは 0xc000007b です。いくつかのアプリケーションやゲームを実行しようとすると、このエラー コードが突然表示され、正常に起動できなくなります。では、この問題をどのように解決すればよいでしょうか?まず、エラー コード 0xc000007b の意味を理解する必要があります。このエラー コードは通常、1 つ以上の重要なシステム ファイルまたはライブラリ ファイルが欠落している、破損している、または正しくないことを示します。

MySQL インストール時に中国語が文字化けする一般的な原因と解決策 MySQL は一般的に使用されるリレーショナル データベース管理システムですが、使用中に中国語の文字化けの問題が発生し、開発者やシステム管理者を悩ませることがあります。中国語の文字化けの問題は、主に文字セットの設定が間違っていることや、データベースサーバーとクライアントの間で文字セットが一致していないことが原因で発生します。この記事では、MySQL インストールにおける中国語の文字化けの一般的な原因と解決策を詳しく紹介し、誰もがこの問題をよりよく解決できるようにします。 1. よくある理由: 文字セットの設定
