もう 1 つの一般的な try-catch 適用可能なシナリオはコールバックです。 。コールバック関数のコードは私たちの制御の範囲外であるため、コードの品質や、例外をスローする他の API が呼び出されるかどうかについてはわかりません。コールバックを呼び出した後の他のコードがコールバック エラーによって実行できなくなるのを防ぐために、呼び出しを try-catch に戻す必要があります。
reportError({
message: errorMessage,
script: scriptURI,
line: lineNumber
});
賢く、window.addEventListener または window.attachEvent を使用して window.onerror をリッスンしないように注意してください。多くのブラウザは window.onerror のみを実装するか、window.onerror の実装のみが標準です。ドラフト標準でも window.onerror が定義されていることを考慮すると、window.onerror をそのまま使用できます。
属性の損失
キャプチャされた例外を収集し、クエリと分析のためにサーバー側のストレージにバッチで送信するための reportError 関数があるとします。では、どのような情報を収集したいのでしょうか?さらに役立つ情報には、エラー タイプ (name)、エラー メッセージ (message)、スクリプト ファイル アドレス (script)、行番号 (line)、列番号 (column)、およびスタック トレース (stack) が含まれます。 try-catch を通じて例外がキャッチされた場合、この情報は Error オブジェクト (主流のブラウザでサポートされている) にあるため、reportError もこの情報を収集できます。しかし、window.onerror を通じてキャプチャされた場合、このイベント関数には 3 つのパラメーターしかないことがわかります。そのため、これら 3 つのパラメーター以外の情報は失われます。
シリアル化されたメッセージ
Error オブジェクトが自分で作成された場合、error.message は自分で制御されます。基本的に、error.message に何を入力しても、window.onerror の最初のパラメータ (メッセージ) になります。 (ブラウザーは実際には、「Uncaught Error: 」プレフィックスを追加するなど、わずかな変更を加えます。) したがって、必要なプロパティ (JSON.Stringify など) をシリアル化し、それらを error.message に保存してから読み取ることができます。 in window.onerror それを取り出して逆シリアル化するだけです。もちろん、これは自分で作成した Error オブジェクトに限定されます。
5 番目のパラメータ
ブラウザの製造元も、window.onerror を使用するときに誰もが直面する制限を認識しているため、window.onerror に新しいパラメータを追加し始めました。行番号のみで列番号がまったく対称的ではないように見えることを考慮して、IE はまず列番号を追加し、それを 4 番目のパラメーターに入れます。しかし、誰もが完全なスタックを取得できるかどうかのほうを心配しているため、Firefox はスタックを 5 番目のパラメータに入れる方がよいと述べています。しかし、Chrome は、Error オブジェクト全体を 5 番目のパラメータに入れる方がよいと述べています。カスタム プロパティを含め、必要なプロパティを読み取ることができます。その結果、Chrome はより高速に動作し、Chrome 30 に新しい window.onerror シグネチャが実装され、それに応じて標準ドラフトが作成されることになりました。
window.onerror = function(
errorMessage,
scriptURI,
lineNumber,
columnNumber,
error
) {
if (error) {
reportError(error) > } else { reportError( {
メッセージ: errorMessage,
スクリプト: scriptURI,
行: lineNumber,
列: 列番号
});
属性の正規化
前に説明した Error オブジェクトの属性の名前は Chrome の命名方法に基づいています。ただし、Chrome では、ブラウザごとにスクリプト ファイルのアドレスなど、Error オブジェクトの属性に名前が付けられます。これはスクリプトと呼ばれますが、Firefox ではファイル名と呼ばれます。したがって、Error オブジェクトを正規化する、つまり、さまざまな属性名を統一された属性名にマップするための特別な関数も必要です。具体的な方法については、こちらの記事をご覧ください。ブラウザの実装は更新されますが、このようなマッピング テーブルを手動で維持することはそれほど難しくありません。
スタック トレースの形式に似ています。この属性は、例外情報のスタックをプレーン テキスト形式で保存します。各ブラウザで使用されるテキスト形式が異なるため、プレーン テキストから各フレームの機能を抽出するための正規表現を手動で管理する必要もあります。識別子)、ファイル(スクリプト)、行番号(line)、列番号(column)。
セキュリティ制限
「スクリプト エラー。」というメッセージが表示されるエラーが発生したことがある場合は、これが実際には、さまざまな元のスクリプト ファイルに対するブラウザの応答であることが理解できるでしょう。このセキュリティ制限の理由は次のとおりです。ユーザーがログインした後にオンライン銀行から返される HTML が匿名ユーザーに表示される HTML と異なると仮定すると、サードパーティの Web サイトはオンライン銀行の URI をスクリプトに組み込むことができます。 src 属性。もちろん、HTML は JS として解析できないため、ブラウザは例外をスローし、サードパーティ Web サイトは例外の場所を解析することでユーザーがログインしているかどうかを判断できます。このため、ブラウザーは、「スクリプト エラー」のような未変更のメッセージが 1 つだけ残り、他のすべての属性が消えるまで、さまざまなソースからのスクリプト ファイルによってスローされたすべての例外をフィルター処理します。
一定規模の Web サイトでは、スクリプト ファイルが異なるソースを持つ CDN に配置されるのが通常です。独自の小規模な Web サイトを構築する場合でも、jQuery や Backbone などの一般的なフレームワークがパブリック CDN 上のバージョンを直接参照して、ユーザーのダウンロードを高速化できるようになりました。したがって、このセキュリティ制限は何らかの問題を引き起こし、Chrome と Firefox から収集した例外情報が役に立たない「スクリプト エラー」になってしまいます。
CORS
この制限を回避するには、スクリプト ファイルとページ自体のオリジンが同じであることを確認するだけです。しかし、CDN によって高速化されていないサーバーにスクリプト ファイルを置くと、ユーザーのダウンロード速度は遅くなりませんか? 1 つの解決策は、スクリプト ファイルを CDN に配置し続け、XMLHttpRequest を使用して CORS 経由でコンテンツをダウンロードし、それをページに挿入するための <script> タグを作成することです。ページに埋め込まれているコードは、もちろん同じソースからのものです。 <br><br>これは簡単そうに見えますが、実装するには詳細がたくさんあります。簡単な例を使用してみましょう: <br><div class="codetitle">
<span><a style="CURSOR: pointer" data="8772" class="copybut" id="copybut8772" onclick="doCopy('code8772')"><u>コードをコピー </u></a></span> コードは次のとおりです: </div>
<div class="codebody" id="code8772"> <br><script src ="http://cdn.com/step1.js"></script>
<script> <br>(関数 step2() {})(); ; <br><script src="http://cdn.com/step3.js"></script>
このステップ 1、ステップ 2、ステップ 3 がある場合は、誰もが知っています。依存関係がある場合は、この順序に厳密に従う必要があります。そうしないと、エラーが発生する可能性があります。ブラウザは step1 と step3 のファイルを並行してリクエストできますが、実行中の順序は保証されます。 XMLHttpRequest を通じて step1 と step3 のファイルの内容を自分で取得する場合は、それらの順序が正しいことを確認する必要があります。さらに、ステップ 2 を忘れないでください。ステップ 2 は、ステップ 1 が非ブロック方式でダウンロードされるときに実行できるため、ステップ 2 に手動で介入して、ステップ 1 が完了するまで待機させてから実行する必要があります。
Web サイト上のさまざまなページに <script> タグを生成するためのツール セットがすでにある場合は、このツール セットを調整して <script> タグを変更する必要があります。 <br><br><div class="codetitle">
<span>コードをコピーします<a style="CURSOR: pointer" data="91019" class="copybut" id="copybut91019" onclick="doCopy('code91019')"><u></u> コードは次のとおりです。</a></span> </div><script<div class="codebody" id="code91019">scheduleRemoteScript('http: //cdn.com/step1.js'); <br></script>
scheduleInlineScript(関数 step2() {})() ;
});
<script> <br>scheduleRemoteScript('http://cdn.com/step3.js'); <br><br><br>scheduleRemoteScript とscheduleInlineScript という 2 つの関数を実装し、外部スクリプト ファイルを参照する最初の <script> タグの前にこれらの関数が定義されていることを確認する必要があります。その後、残りの <script> タグが上記のように書き換えられます。形状。元々すぐに実行された step2 関数が、より大きなコード関数に配置されていることに注意してください。コード関数は実行されません。これは単なるコンテナであるため、ステップ 2 の元のコードはエスケープせずに保持できますが、すぐには実行されません。 <br><br>次に、アドレスに基づいてscheduleRemoteScriptによってダウンロードされたファイルコンテンツと、scheduleInlineScriptによって直接取得されたコードが正しい順序で次々に実行できることを保証する完全なメカニズムを実装する必要があります。詳細なコードはここでは説明しませんので、興味があればご自身で実装してください。 <br>行番号逆チェック <br><br>CORS を通じてコンテンツを取得し、ページにコードを挿入すると、セキュリティ制限を突破できますが、行番号の競合という新たな問題が発生します。もともと、error.script は一意のスクリプト ファイルを見つけるために使用でき、error.line は一意の行番号を見つけるために使用できました。さて、これらはすべてページに埋め込まれたコードであるため、error.script では複数の <script> タグを区別することはできません。そのため、各 <script> タグ内の行番号は 1 から始まります。エラーが発生したソース コードの場所を特定するために使用されます。 <br><br>行番号の競合を避けるために、各 <script> タグ内の実際のコードで使用される行番号の範囲が互いに重ならないように、一部の行番号を無駄にすることができます。たとえば、各 <script> タグの実際のコードが 1000 行以下であると仮定すると、最初の <script> タグのコードを 1 ~ 1000 行とし、2 番目のコードを<script> タグは 1001 ~ 2000 行目 (その前に 1000 行の空白行を挿入) を占め、3 番目の <script> タグ内のコードは 2001 ~ 3000 行目 (その前に 2000 行の空白行を挿入) を占めます。次に、data-* 属性を使用してこの情報を記録し、簡単に取得できるようにします。 <BR><div class="codetitle"><span><a style="CURSOR: pointer" data="17722" class="copybut" id="copybut17722" onclick="doCopy('code17722')"><U>コードをコピー コードは次のとおりです:</script>