みなさんこんにちは。私は次男です。 最初の2つの記事が公開された後、何人かのネットユーザーが裏で私にいくつかの質問をしたので、それらをまとめて要約しました。この記事はネットユーザーの質問への回答であると同時に、前の2つの記事の補足とレビューでもあります。 最初の 2 つの記事へのリンクは次のとおりです。 Node.js のコアイベントループのグラフィカルな説明 マルチグラフ分析式 async=Promise+Generator+Automatic Executor 図1: 非同期関数のコード例 質問 0 : 前の記事で説明したジェネレーターと自動実行プログラムは、異なるスレッドで実行されますか? 回答 0 : ジェネレーターとエグゼキューターは両方とも、JS コードを実行するメイン スレッドであるイベント ループ スレッドで実行されます。もう一度強調しておきますが、これらは 2 つのスレッドで実行されているわけではありません。 Node.js公式サイトのevent-loopの説明をもう一度見てみましょう。重要な点を強調します。JS コードはシングルスレッド方式で実行されます。 イベントループにより、 JavaScript はシングル スレッドであるにもかかわらず、 可能な限り操作をシステムカーネルにオフロードすることで、 Node.js は非ブロッキングI / O 操作を実行できるようになります。 質問 1 : ステートメント await p は非同期要求を生成しますか? 回答 1 : いいえ、そうではありません。 await は、p の状態が保留から解決に変化するか、保留から拒否に変化するかを待機するだけです。 質問 2 : 非同期リクエストはいつ生成されますか? 回答 2 : setTimeout が実行されると Promise executor で生成されます。 次のテキストでは、new Promise() を呼び出すときに渡されるコールバック (resolve, reject)=>{ /* your code */ は、executor と呼ばれます。パラメータresolveとrejectはPromise自体によって実装されます。このエグゼキュータは、new Promise() が呼び出されるとすぐに実行されることに注意してください。 executor で fs.read(fd[, options], callback) のようなステートメントを実行すると、同様に、fs.read() が呼び出されたときに非同期リクエストが生成されます。 質問3 :pの状態が変化した後、resolve(200)を通過した200が変数resの評価結果になるのはなぜですか? 回答 3 : これが、await の背後にある実装原則を理解する必要がある理由です。図 2 と 4 を参考にしてこれを確認してみましょう。 図 2 に示すように、まず async 関数がジェネレーター関数に変換されます。ただし、ジェネレーター関数はそれ自体では自動的に実行できないため、自動実行プログラムと組み合わせて実行する必要があります。オートマトンとは愛情深い母親のようなもので、ジェネレーターとは純真な幼児のようなものだ。子どもは一定の距離を歩くたびに立ち止まり、後ろからついてくる母親を振り返り、母親から励ましやご褒美をもらって次の目標へと進んでいきます。 図2: 非同期関数をジェネレータ関数に変換する例 図 4 を説明する前に、yield 式と yield ステートメントという 2 つの重要な概念を確認する必要があります。図3に示すように:
図 3 には、ジェネレーター関数の実行の一時停止ポイント、つまり yield 式が評価された後、yield ステートメントが返される前という重要なポイントも示されています。 図3: 収量式と収量ステートメントの比較 質問 3 にもっとわかりやすく答えるために、次兄が図 4 を描いてくれました。 ① このステップでは、エグゼキュータを介してジェネレータの呼び出しを開始します。 ② ここで実際のジェネレータの呼び出しが発生しますが、ジェネレータ関数はステップ ② では何も行わず、すぐにイテレータを返します。 ③ここから自動アクチュエータは駆動ジェネレータモードに入ります。 ③ このステップでは、g.next() が初めて実行されるときに値を挿入できないため、パラメータデータに値を割り当てません。 ④ このステップで g.next() が呼び出されるたびに、ジェネレーターは yield で最後に一時停止した位置から実行を開始し、再び yield に遭遇するまで実行されます。 ⑤ したがって、g.next() の最初の呼び出しにより、左のジェネレーター関数は関数の先頭から yield に遭遇するまで実行されます。 ⑤でマークされたコード実行プロセスでは、実際に Promise オブジェクトが作成され、Promise executor に 1 秒のタイマーが設定されていることがわかります。このエグゼキュータは Promise オブジェクトが作成されると同時に実行されますが、⑦のコードは 1 秒後まで実行されないことに注意してください。 ⑥ジェネレータ関数が一時停止する前に、yield式の評価結果が{value:xxx, done:xxx}を通じてg.next()の呼び出し元に返されます。これは右図の④の位置です。 したがって、右の図の位置 ④ にある変数 result は { value: p, done: false} であると推測できるはずです。ここでpは⑤の実行中に生成されたPromiseオブジェクトです。 このようにして、Promise オブジェクトはジェネレーター関数と自動エグゼキューターの間を流れます。それはとても独創的なプロセスです。 そうすれば、右側のステートメント result.value.then(callback) を見ても混乱することはありません。これは Promise の標準的な使用法です。 p の状態が解決されると、当然 ⑧ のコールバックが実行される機会が得られます。 ⑦ 1秒があっという間に経過し、resolve(200)が実行できるようになります。これを実行するとpの状態が解決されるため、⑧で辛抱強く待機していたコールバックが動作を開始します。 ⑧ はい、この時点でデータの値は200です。 Promise の使用に慣れている場合、これは非常に自然なことです。 ⑨ 自動実行プログラムは next(data) を再度実行します。しかし今回は実際のパラメータ200が渡されます。そのため、今回は④で実行されるコードは g.next(200) になります。 ⑩自動実行プログラムはg.next(200)を実行し、必然的にジェネレータ関数を開始して前進を続けます。 前回ジェネレータ関数がどこで停止したかを覚えていますか?はい、左側の⑤の矢印の位置です。ジェネレータ関数が再開したときに最初に行うことは、yield ステートメントを評価することです。 g.next() のように駆動すると、yield ステートメントは undefined を返します。しかし今回は g.next(200) を実行する点が異なります。非常に巧妙なことに、 next() に渡された引数 200 は、 yield ステートメントの戻り値として左変数 res に割り当てられます。 図4: ジェネレーター + オートメーターの詳細 図 1 のサンプル コードをもう一度見て、要約してみましょう。
図5: 図1と同じ |
<<: VMware は、ワークロードのパフォーマンスを向上させ、企業の TCO を削減する新しいコンピューティングおよびストレージ ソリューションをリリースしました。
>>: Prometheus、Istio、Hpa、Keda、Karpenter をベースにした K8s アプリケーションとノードの弾力性の実装
クラウド バックアップは、企業がファイルとサーバーの最新のコピーを作成できるようにするサービスです。...
「ダブル11」プロモーション戦争における電子商取引業界の競争は、価格と物流の競争からトラフィックをめ...
元宵節にGoogleから弊社に「大きなプレゼント」が届きました。まさに青天の霹靂でした。Google...
長い8日間の休暇が終わりました。この間、私たちはとても楽しい時間を過ごしました... 戻ってきたとき...
ホームページ運営は非常に知的な仕事ですが、ホームページ企画はさらに知的な職業であり、知恵をフル活用し...
現在から 4 月 10 日まで、edgenat はすべての VPS に対して月額支払いで 30% オ...
ポータルサイトを開くと、サイト内検索、つまりサイト内検索があり、ユーザーが必要な情報をすばやく見つけ...
1930年代には『ウォータールー橋』『風と共に去りぬ』『ミッキーマウスとドナルドダック』など優れた映...
かつては、多くの人がビッグデータとクラウドコンピューティングを別々のテクノロジーとして見ていました。...
多くのウェブマスターにとって、ウェブサイトがBaiduにKまたは降格された後、一定期間公開されないこ...
ウェブサイトの最適化といえば、編集者は重要な外部リンクの構築について言及しなければなりません。外部リ...
価格競争ですでに赤字に陥っていた国内クラウドベンダーは、再び大きな危機に直面している。最近、2017...
2021年11月3日、「デジタルと現実の融合、新たなチャンスの爆発」をテーマにしたテンセントデジタル...
[要点] この記事では、Douban ユーザーの視点から、Douban がユーザーにとってどのように...
BilibiliとXigua Videoの間で、新たな大Vsの攻防戦が始まった。表面的には、この競争...