JS関数の呼び出し、適用、バインドの超詳細な方法

JS関数の呼び出し、適用、バインドの超詳細な方法

JS 関数呼び出し、適用、バインドメソッド

1. call() メソッド

call()メソッドを呼び出すと、ターゲット関数がすぐに実行され、関数内のthisの参照が変更されます。これはメソッドの最初のパラメータを指し、 1 つずつリストされたパラメータは、ターゲット関数のパラメータとして 1 つずつ渡されます。

/* 通常モード */

obj = {
  合計(a, b) {
    console.log(これ)
    a + bを返す
  }
}

// sum 関数の apply メソッドと bind メソッドを実行します。出力される内容は以下のようになります。obj.sum.call() // ウィンドウを印刷
obj.sum.call(undefined, 1, 2) // 印刷ウィンドウ
obj.sum.call(null, 1, 2) // 印刷ウィンドウ

/* 厳密モード */
「厳密な使用」

// sum関数のapplyメソッドとbindメソッドを実行し、以下のように出力します。obj.sum.call() // undefinedを出力します。
obj.sum.call(undefined, 1, 2) // undefined と表示される
obj.sum.call(null, 1, 2) // null を出力します

1. call() メソッドのシミュレーション実装

要点:

  • myCall()メソッドが Function プロトタイプ オブジェクトに追加されます。ターゲット関数がこのメソッドを呼び出すと、myCall() メソッド内の this がターゲット関数を指します。
  • ターゲット関数をコンテキスト オブジェクトのメソッドとして実行し、ターゲット関数内の this がコンテキスト オブジェクトを指すようにします。
  • コンテキストオブジェクトからターゲット関数を削除します
  • スプレッド演算子...を使用して、ターゲット関数に渡されたパラメータを処理します。

call()、apply()、bind() メソッドのシミュレーション実装では、最初のパラメータが渡されないか、undefined または null が渡された場合、ここでは JS 通常モードと厳密モードで統一された処理が行われ、つまり、対象関数内の this は window オブジェクトを指します。

コードは次のとおりです。

Function.prototype.myCall = function (コンテキスト、...引数) {
  コンテキスト === 未定義 || コンテキスト === null の場合 {
    コンテキスト = ウィンドウ
  }
  // 次の行はコアコードです context.fn = this
  const 結果 = context.fn(...args)
  コンテキストを削除する.fn
  結果を返す
}

obj1 = {
  基本数: 1,
  合計(a, b) {
    console.log(これ)
    this.basicNum + a + b を返す
  }
}
obj2 = {
  基本数: 9
}
コンソールログ(obj1.sum.call(obj2, 2, 3)) // 14
コンソールログ(obj1.sum.myCall(obj2, 2, 3)) // 14

2. apply() メソッド

apply()メソッドを呼び出すと、ターゲット関数が直ちに実行され、関数内のthisの参照が変更されます。これはメソッドの最初のパラメータを指します。2 番目のパラメータは、パラメータ配列または引数オブジェクトです。各配列要素または引数オブジェクトによって表される各パラメータは、ターゲット関数のパラメータとして 1 つずつ渡されます。

1. apply() メソッドのシミュレーション実装

要点:

  • myApply()メソッドが Function プロトタイプ オブジェクトに追加されます。ターゲット関数がこのメソッドを呼び出すと、myApply() メソッド内の this がターゲット関数を指します。
  • ターゲット関数をコンテキスト オブジェクトのメソッドとして実行し、ターゲット関数内の this がコンテキスト オブジェクトを指すようにします。
  • コンテキストオブジェクトからターゲット関数を削除します
  • スプレッド演算子...を使用して、ターゲット関数に渡されたパラメータを処理します。

コードは次のとおりです。

Function.prototype.myApply = 関数 (コンテキスト、引数) {
  コンテキスト === 未定義 || コンテキスト === null の場合 {
    コンテキスト = ウィンドウ
  }
  // 次の行はコアコードです context.fn = this
  const 結果 = context.fn(...args)
  コンテキストを削除する.fn
  結果を返す
}

コンソール.log(obj1.sum.apply(obj2, [2, 3])) // 14
コンソール.log(obj1.sum.myApply(obj2, [2, 3])) // 14

3. bind() メソッド

  • bind()メソッドを呼び出すと、新しい関数 (ターゲット関数のコピー) が返されます。関数内のthisメソッドの最初のパラメータを指し、1 つずつリストされたパラメータは、ターゲット関数のパラメータとして 1 つずつ渡されます。その後に新しい関数を実行することは、ターゲット関数を実行することと同じです。
  • bind()メソッドは関数のカリー化を実装しているため、ターゲット関数にパラメータを 2 回渡すことができます。最初のパラメータは bind() メソッドの最初のパラメータの後にリストされ、2 番目のパラメータは新しい関数にリストされます。

1. bind() メソッドのシミュレーション実装

要点:

  • myBind()メソッドが Function プロトタイプ オブジェクトに追加されます。ターゲット関数がこのメソッドを呼び出すと、myBind() メソッド内の this がターゲット関数を指します。
  • ターゲット関数をコンテキスト オブジェクトのメソッドとして実行し、ターゲット関数内の this がコンテキスト オブジェクトを指すようにします。
  • コンテキストオブジェクトからターゲット関数を削除します
  • スプレッド演算子...を使用して、ターゲット関数に渡される最初のパラメータと後続のパラメータを処理します。

コードは次のとおりです。

Function.prototype.myBind = function (context, ...initArgs) {
  コンテキスト === 未定義 || コンテキスト === null の場合 {
    コンテキスト = ウィンドウ
  }
  // この値をキャッシュする const _this = this
  関数を返す (...引数) {
    // 次の行はコアコードです context.fn = _this
    const 結果 = context.fn(...initArgs, ...args)
    コンテキストを削除する.fn
    結果を返す
  }
}

コンソール.log(obj1.sum.bind(obj2, 2)(3)) // 14
コンソール.log(obj1.sum.myBind(obj2, 2)(3)) // 14

IV. 結論

3 つの方法の類似点と相違点:

類似点:
どちらも、ターゲット関数の実行時に内部 this ポイントを変更できます。メソッドの最初のパラメーターは、関数の実行時に内部 this 値を指定するために使用されます。ターゲット関数に任意の数のパラメーターを渡すことがサポートされています。メソッドの最初のパラメーターに値が渡されないか、undefined または null が渡された場合、JavaScript の通常モードでは、ターゲット関数内の this は window オブジェクトを指し、厳密モードでは、それぞれ undefined または null を指します。

違い:
apply() メソッドは 2 つのパラメータを受け入れますが、call() メソッドと bind() メソッドは複数のパラメータを受け入れます。
apply() メソッドがターゲット関数にパラメータを渡す場合、メソッドの 2 番目のパラメータとしてパラメータ配列または引数オブジェクトを渡すだけで済みますが、call() メソッドと bind() メソッドでは、メソッドのパラメータの後にパラメータを 1 つずつリストする必要があります。
call() メソッドと apply() メソッドが呼び出されると、ターゲット関数はすぐに実行されますが、bind() メソッドは実行されません。bind() メソッドは、ターゲット関数のコピーである新しい関数を返します。関数内の this は、bind() メソッドの最初のパラメータを指します。その後に新しい関数を実行することは、ターゲット関数を実行することと同じです。
bind() メソッドのみが関数カリー化を実装しているため、ターゲット関数にパラメータを 2 回渡すことができます。

上記は、JS 関数の呼び出し、適用、バインドの超詳細なメソッドの詳細です。JS 関数の呼び出し、適用、バインドメソッドの詳細については、123WORDPRESS.COM の他の関連記事に注目してください。

以下もご興味があるかもしれません:
  • JavaScript 関数型プログラミングの基礎
  • JS関数の継承について学ぶ記事
  • JavaScript 基礎シリーズ: 関数とメソッド
  • JavaScript 関数呼び出し、適用、バインド メソッドのケース スタディ
  • JavaScript の矢印関数と通常の関数の違いの詳細な説明
  • JavaScript の知識: コンストラクタも関数である
  • JavaScriptのスリープ関数の使用
  • JavaScript における変数と関数の昇格の詳細な例
  • JavaScript の 50 以上のユーティリティ関数の概要
  • 集める価値のある 15 個の JavaScript 関数

<<:  よく使われるLinuxコマンド「ll」が無効、またはコマンドが見つからないという問題を解決します

>>:  MySQL のスロークエリの方法と例

推薦する

VMware仮想マシンブリッジによるインターネット相互接続を実現する方法

VMware をインストールして新しい仮想マシンを作成したら、オプション バーの [編集] - [仮...

Unix/Linuxフォークの隠れたオーバーヘッド

目次1. フォークの起源2. 初期のUNIXオーバーレイ技術3. UNIXに導入される前のフォークの...

vue3.0 プロジェクトを素早く構築するための手順を完了する

目次1. 3.0をより適切にサポートするには、vue/cliのバージョンが4.5.0以上であることを...

MySQLサービスを開くおよび閉じる2つの方法

方法1: cmdコマンドを使用するまず、DOS ウィンドウを開き、スタート、実行、cmd と入力しま...

空のパスがページのパフォーマンスに与える影響に対する解決策

数日前、Google Reader で Yu Bo さんが共有した投稿「空のパスがページのパフォーマ...

良い広告にはどのような特徴が必要ですか?

広告業は人間であることに似ていると言う人がいます。これは本当です。優れた広告には、優れた人間と同じよ...

MySQLのCOUNT(*)のパフォーマンスについてお話しましょう

序文基本的に、職場のプログラマーは、count(*)、count(1)、または count(prim...

node_modulesを削除して再インストールする方法

目次ステップ1: プロジェクトをインストールするディレクトリにnode_modulesをインストール...

ドラッグアンドドロップでVueユーザーインターフェースを生成する方法

目次序文1. 技術原理1.1 レイアウト1.2 コンポーネント1.3 ステータス1.4 イベント1....

docker compose を使用して FastDfs ファイル サーバーをインストールする詳細な例

ドッカーの作成 バージョン: '2' サービス: fastdfsトラッカー: ホスト...

Docker を使用してイメージをローカルにパッケージ化してデプロイする方法

初めてDockerを使用してイメージをローカルにパッケージ化してデプロイするまず、私のラップトップシ...

リンク内の href=# はどういう意味ですか?

現在のページへのリンク。 -------------------一般的な使用法は次のとおりです。 &...

VMware Workstation Pro でサーバー仮想マシンを構築する (グラフィック チュートリアル)

私が使用している VMware Workstation Pro のバージョンは次のとおりです。 1....

Linux で docker-compose を使用したソフトウェア構成の詳細な説明

序文この記事では、docker-compose の構成をいくつか紹介します。これらを参考にして、独自...

Vueグローバルカスタム命令の実践 モーダルドラッグ

目次背景実装のアイデア成果を達成する背景最近取り組んでいるプロジェクトは、Vue2 で構築されたプロ...