JS での new の手書き実装

JS での new の手書き実装

⚠ 前提条件:

  • プロトタイプとプロトタイプチェーンを理解する
  • このバインディングを理解する

1 新しいオペレータの紹介

MDN ドキュメント: new演算子は、ユーザー定義のオブジェクト タイプのインスタンス、またはコンストラクターを持つ組み込みオブジェクトのインスタンスを作成します。

クラス Person {
    コンストラクタ(名前) {
        this.name = 名前;
    }
}
// カスタムオブジェクトタイプのインスタンスを作成する const person = new Person('Xiao Ming')
// コンストラクタを使用して組み込みオブジェクトのインスタンスを作成する const date = new Date()


newの役割:オブジェクトのインスタンスを作成する

2 新しいものは何をしましたか?

上で、 newの機能はオブジェクトのインスタンスを作成することだと述べましたが、インスタンスはどのように作成され、内部では何が行われるのでしょうか。

new Person() を例に挙げてみましょう。これを実行すると、次のことが起こります。

空のシンプルなJSオブジェクトを作成する

定数オブジェクト = {}


このオブジェクトに属性__proto__,を追加し、その属性をコンストラクタのプロトタイプオブジェクトにリンクします。

obj.__proto__ = Person.プロトタイプ


コンストラクタPersonを呼び出し、 this新しく作成されたオブジェクトobjにバインドします。

Person.apply(オブジェクト)


コンストラクタが明示的にオブジェクトを返さない場合は、新しく作成されたオブジェクト、つまりobjが返されます。

3 新しい演算子の実装をシミュレートする

上で述べたように、 new演算子は 4 つのことを行います。関数を使用して、これらの 4 つの手順に従ってnewの実装をシミュレートしてみましょう (インタビュー用の手書きコード)

const _new = 関数(コンストラクタ、...引数) {
    定数オブジェクト = {}
    obj.__proto__ = コンストラクタ.prototype
    const res = コンストラクタ.apply(obj, args)
    // この手順については「補足」で詳しく説明します return res instanceof Object ? res : obj
}

コードは非常にシンプルで、上記の4つの手順に従って段階的に記述するだけです。

4 補足

ES5にはObject.createメソッドが用意されており、これを使用するとオブジェクトを作成し、新しいオブジェクトの__proto__プロパティが既存のオブジェクトを指すようにすることができます。

この方法を使ってステップ1と2を組み合わせることができます

const obj = Object.create(コンストラクタ.プロトタイプ)
// const obj = {} と同等
obj.__proto__ = コンストラクタ.prototype

ステップ4では、

  • コンストラクターが明示的にreturnない場合 (通常)、 personは新しく作成されたオブジェクトobj
  • コンストラクタが1や「abc」などのオブジェクトを返さない場合、 person新しく作成されたオブジェクトobjのままです。
関数Person() {
    ...
   戻り値 1
}

コンストラクタが明示的にオブジェクトを返す場合(例: {}function() {}

この場合、 person新しく作成されたオブジェクトobjではなく、明示的にreturnオブジェクトになります。

関数Person() {
  // 関数もオブジェクトです return function() {}
}

したがって、_new 関数の最後のコード行は次のようになります。

res インスタンスオブオブジェクトを返します? res : obj


注:シミュレートされた関数_newに渡されるパラメータは、クラスではなくコンストラクタのみです。

class Animal { ...}_new(Animal) // エラーを報告します: クラスコンストラクタ Animal は 'new' なしでは呼び出せません // クラスは new を使用してのみ作成できます

JS での new の手書き実装に関するこの記事はこれで終わりです。JS でのより関連性の高い new コンテンツについては、123WORDPRESS.COM の以前の記事を検索するか、以下の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM をよろしくお願いいたします。

以下もご興味があるかもしれません:
  • JSコンストラクタとインスタンス化およびプロトタイプ導入の関係
  • JavaScriptプロトタイプと例の詳細な説明
  • Js の継承とプロトタイプチェーンを理解するのに役立つ記事
  • JavaScriptプロトタイプチェーン図のまとめと実践
  • JS の new 関数の詳細な説明
  • JavaScript のコンストラクター、プロトタイプ、プロトタイプ チェーン、new についてどれくらい知っていますか?

<<:  Html、sHtml、XHtml の違いのまとめ

>>:  MySQL InnoDB ストレージエンジンのメモリ管理の詳細な説明

推薦する

数十億のデータに対するMySQLページングの最適化に関する簡単な説明

目次背景分析するデータシミュレーション1. 従業員テーブルと部門テーブルの2つのテーブルを作成します...

Vue で SVG アイコンを導入する 2 つの方法

Vue で SVG アイコンを導入する方法Vue で svg アイコンを導入する方法 1インストール...

mysql 更新ケース更新フィールド値が固定されていない操作

特定のデータの一括更新処理において、特定のステータスが固定値に更新されるなど、更新するフィールドの値...

CSS画像結合技術(スプライト画像)の詳しい説明

CSS画像結合技術1. 画像のステッチ画像ステッチング技術は、個々の画像を収集する技術です。画像の多...

MySQL の group by と having の詳細な説明

GROUP BY 構文を使用すると、指定されたデータ列の各メンバーに従ってクエリ結果をグループ化して...

純粋な CSS でフォ​​ーム検証を実装するためのサンプル コード

日常業務において、フォームの検証は非常に一般的な設計要件です。ログイン ボックスや登録ボックス、アン...

MIME エンコーディングの概要 (オンライン情報と実際の経験から統合)

1. MIME: 多目的インターネットメール拡張インペリアル カレッジ オブ コンピュータ オンラ...

MySQL ストアド プロシージャで case ステートメントを使用する詳細な例

この記事では、例を使用して、MySQL ストアド プロシージャでの case ステートメントの使用方...

HTML タグのカスタム属性に関する質問

以前の開発では、クラス、名前などの HTML のデフォルト属性を使用していました。 Huawei社の...

DockerでNginxサーバーを作成する方法

動作環境: MAC Docker バージョン: Docker version 17.12.0-ce,...

MySQL テーブルを削除する際の I/O エラーの原因分析と解決方法

問題現象最近、sysbench を使用して MySQL をテストしました。テストに長い時間がかかった...

W3C チュートリアル (8): W3C XML スキーマのアクティビティ

XML スキーマは、DTD に代わる XML ベースのものです。 XML スキーマは、DTD に代わ...

CentOS 8 Zabbix 4.4 インストール ガイド

Zabbix サーバー環境プラットフォームZABBIX バージョン 4.4セントOS8 MySQL ...

Dockerコンテナの起動失敗を解決する方法

質問: コンピュータを再起動した後、docker の mysql コンテナを再起動できません。原因が...