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 ストレージエンジンのメモリ管理の詳細な説明

推薦する

CSS3 アドバンス LESS で星空アニメーションを実装するサンプルコード

この記事では、星空アニメーションを実現するための高度な CSS3 LESS のサンプルコードを次のよ...

MySQL MyISAM と InnoDB の違い

違い: 1. InnoDB はトランザクションをサポートしていますが、MyISAM はサポートしてい...

Vue の下部ナビゲーション バー TabBar を実装するための非常に詳細なチュートリアル

目次プロジェクト紹介:プロジェクトディレクトリ: TabBar 効果のプレビュー: TabBar 実...

現在使用されている設定ファイル my.cnf を表示する mysql メソッド (推奨)

my.cnfは、MySQL の起動時に読み込まれる設定ファイルです。通常は MySQL インストー...

TypeScript をインストール、使用、自動コンパイルする方法に関するチュートリアル

1. TypeScriptの紹介前回の記事ではTypeScriptのインストール、使い方、自動コンパ...

HTML構造化実装方法

DIV+css構造 CSSレイアウトを学んでいますか?まだ純粋な CSS レイアウトを完全に習得でき...

MySQLカバーインデックスの使用例

カバーインデックスとは何ですか?クエリで使用されるすべてのフィールドを含むインデックスを作成すること...

CSS3 グラデーション背景の互換性の問題

グラデーションの背景色を作成するときは、 linear-gradient() 関数を使用して線形グラ...

K3s 入門ガイド - Docker で K3s を実行するための詳細なチュートリアル

k3dとは何ですか? k3d は、Docker で K3s クラスターを実行するための小さなプログ...

MYSQL の binlog 最適化に関する考察の要約

質問質問 1: トランザクションをコミットするときに REDO ログをフラッシュすることによって発生...

Nginx サービスを使用してサブドメイン環境を構築し、2D マップの読み込みパフォーマンスを向上させる方法を説明します。

1. 背景最近、友人が大規模なマップの読み込みが遅いという問題に遭遇しました。iServer のパ...

クールなページング効果を実現するネイティブJS

この記事では、次のような効果を持つ JS ページング効果の例を紹介します。クールだと思いませんか? ...

sqlite を mysql スクリプトに移行する方法

さっそく、コードを直接投稿します。具体的なコードは次のとおりです。 パーレル # # https:/...

HTML ではスペースはどのように表現されますか (どのような意味ですか)?

Web 開発では、次のような文字によく遭遇します: &nbsp;これは実際には HTML が...

MySQL が innobackupex を使用して接続サーバーをバックアップできない場合の解決策

innobackupex を使用してバックアップする際に MySQL がサーバーに接続できない場合は...