Maison > interface Web > js tutoriel > JavaScript prend le contrôle du nouveau

JavaScript prend le contrôle du nouveau

WBOY
Libérer: 2022-04-08 19:12:39
avant
2267 Les gens l'ont consulté

Cet article vous apporte des connaissances pertinentes sur javascript, qui présente principalement des problèmes liés à new. L'opérateur new crée une instance d'un type d'objet défini par l'utilisateur ou l'un des types d'objet intégrés avec un constructeur. .

JavaScript prend le contrôle du nouveau

【Recommandations associées : Tutoriel vidéo javascript

Avant-propos

Qu'est-ce que le nouveau ? new呢?

new运算符创建一个用户定义的对象类型的实例或具有构造函数的内置对象类型之一。

光看定义还是有几分晦涩,直接看一个具体的例子,来了解一下JavaScript中的new实现的功能。

举个例子

// 现实中瘦不了,但网络中一定要保持苗条
function Thin_User(name, age) {
    this.name = name;
    this.age = age;
}

Thin_User.prototype.eatToMuch = function () {
    // 白日做梦吧,留下肥胖的泪水
    console.log('i eat so much, but i\'m very thin!!!');
}

Thin_User.prototype.isThin = true;

const xiaobao = new Thin_User('zcxiaobao', 18);
console.log(xiaobao.name);   // zcxiaobao
console.log(xiaobao.age);    // 18
console.log(xiaobao.isThin); // true
// i eat so much, but i'm very thin!!!
xiaobao.eatToMuch();
Copier après la connexion

通过上面这个例子,我们可以发现xiaobao可以:

  • 访问到构造函数Thin_User中属性
  • 访问到Thin_User.prototype中属性

描述得更直白一点,new做了这些事:

  • 创建了一个空对象,对象的__proto__->Thin_User.prototype
  • 执行构造函数,并将this指向新对象
  • 返回新对象

补充说明

由于new是关键字,我们无法像模拟数组高阶方法一样覆盖,因此我们写一个函数createObject,来模拟new的效果。使用具体如下:

function Thin_User(name, age) {}

const u1 = new Thin_user(...)
const u2 = createObject(Thin_User, ...a\)
Copier après la connexion

初步模拟

根据上面分析,createObject编写的大致步骤为:

  • 创建一个新对象obj
  • 设置obj.__proto__->constructor.prototype(但JavaScript不推荐直接修改__proto__属性,提供了setPrototypeOf方法来专门修改原型)
  • 使用constructor.call/apply(obj, ...),将属性添加到obj
  • 返回obj

__proto__和prototype,可以看JavaScript之彻底理解原型与原型链
call/apply,可以看JavaScript之手撕call、apply

学习完这些,我们就可以编写第一版代码:

function createObject(Con) {
    // 创建新对象obj
    // var obj = {};也可以
    var obj = Object.create(null);

    // 将obj.__proto__ -> 构造函数原型
    // (不推荐)obj.__proto__ = Con.prototype
    Object.setPrototypeOf(obj, Con.prototype);

    // 执行构造函数
    Con.apply(obj, [].slice.call(arguments, 1));

    // 返回新对象
    return obj;}
Copier après la connexion

返回值效果

众所周知,函数是有返回值的,那构造函数如果有返回值,最终执行new后返回的结果是怎样的那?

返回值为基本类型

假设构造函数返回值为一个基本类型,我们来看一下最后的返回结果:

function Thin_User(name, age) {
    this.name = name;
    this.age = age;
    return 'i will keep thin forever';
}

Thin_User.prototype.eatToMuch = function () {
    console.log('i eat so much, but i\'m very thin!!!');
}

Thin_User.prototype.isThin = true;

const xiaobao = new Thin_User('zcxiaobao', 18);
console.log(xiaobao.name);   // zcxiaobao
console.log(xiaobao.age);    // 18
console.log(xiaobao.isThin); // true
// i eat so much, but i'm very thin!!!
xiaobao.eatToMuch();
Copier après la connexion

最后的返回结果好像受到任何干扰,难道构造函数不会对返回值进行处理吗?

不急,我们来接着测试一下返回值为对象的情况。

返回值为对象

function Thin_User(name, age) {
    this.name = name;
    this.age = age;
    return {
        name: name,
        age: age * 10,
        fat: true
    }
}

Thin_User.prototype.eatToMuch = function () {
    // 白日做梦吧,留下肥胖的泪水
    console.log('i eat so much, but i\'m very thin!!!');
}

Thin_User.prototype.isThin = true;

const xiaobao = new Thin_User('zcxiaobao', 18);
// Error: xiaobao.eatToMuch is not a function
xiaobao.eatToMuch();
Copier après la connexion

当我执行eatToMuch时,控制台直接报错,没有当前函数,于是我打印了xiaobao对象:
JavaScript prend le contrôle du nouveau

发现xiaobao对象的age发生了改变,而且增加了fat

L'opérateur new crée une instance d'un type d'objet défini par l'utilisateur ou de l'un des types d'objet intégrés dotés d'un constructeur.

C'est encore un peu obscur rien qu'en regardant la définition. Regardons un exemple spécifique pour comprendre les fonctions implémentées par new en JavaScript.

Par exemple

function createObject(Con) {
    // 创建新对象obj
    // var obj = {};也可以
    var obj = Object.create(null);

    // 将obj.__proto__ -> 构造函数原型
    // (不推荐)obj.__proto__ = Con.prototype
    Object.setPrototypeOf(obj, Con.prototype);

    // 执行构造函数,并接受构造函数返回值
    const ret = Con.apply(obj, [].slice.call(arguments, 1));

    // 若构造函数返回值为对象,直接返回该对象
    // 否则返回obj
    return typeof(ret) === 'object' ? ret: obj;}
Copier après la connexion
Grâce à l'exemple ci-dessus, nous pouvons constater que xiaobao peut :

  • Accéder au constructeur Thin_User Attributs dans le code>
  • Accédez aux attributs dans Thin_User.prototype
La description est plus simple, new fait ces choses :
  • Créez un objet vide, le __proto__->Thin_User.prototype
  • de l'objet exécute le constructeur et définit this Pointer vers un nouvel objet
  • Renvoyer un nouvel objet

Explication supplémentaire🎜

🎜Puisque new est un mot-clé, nous pouvons' Pour ne pas le remplacer comme pour simuler des méthodes d'ordre supérieur de tableau, nous écrivons donc une fonction createObject pour simuler l'effet de new. L'utilisation spécifique est la suivante : 🎜rrreee🎜🎜Simulation préliminaire🎜🎜🎜Selon l'analyse ci-dessus, les étapes générales pour écrire createObject sont : 🎜
  • Créer un nouvel objet obj
  • Set obj.__proto__->constructor.prototype(🎜Mais JavaScript ne recommande pas de modifier directement l'attribut __proto__ et fournit la méthode setPrototypeOf pour spécifiquement modifier le prototype🎜)
  • Utilisez constructor.call/apply(obj, ...) pour ajouter des propriétés à obj
  • Renvoyer obj
🎜__proto__ et prototype, vous pouvez voir JavaScript pour une compréhension complète des prototypes et des chaînes de prototypes
call/apply code>, vous pouvez voir l'appel de JavaScript et postuler🎜
🎜Après les avoir appris, nous pouvons écrire la première version du code :🎜rrreee🎜🎜Effet de valeur de retour🎜🎜🎜Comme nous tout le monde le sait, les fonctions ont des valeurs de retour. Si le constructeur a une valeur de retour, quel est le résultat renvoyé après l'exécution de new ? 🎜

🎜La valeur de retour est un type de base🎜

🎜En supposant que la valeur de retour du constructeur est un type de base, jetons un œil au résultat de retour final : 🎜rrreee🎜Le résultat de retour final semble ont été interférés de quelque manière que ce soit. Se pourrait-il que le constructeur ne traite pas la valeur de retour ? 🎜🎜Pas pressé, continuons à tester le cas où la valeur de retour est un objet. 🎜

🎜La valeur de retour est un objet🎜

rrreee🎜Quand j'ai exécuté eatToMuch, la console a signalé une erreur directement et il n'y avait aucune fonction actuelle, j'ai donc imprimé xiaobao Objet :
Insérer la description de l'image ici🎜🎜 On constate que le age de l'objet xiaobao a changé, et l'attribut fat a été ajouté, qui est exactement le même comme valeur de retour du constructeur. 🎜🎜Après avoir lu ces deux exemples, vous pouvez en gros clarifier la situation lorsque le constructeur renvoie une valeur : 🎜Lorsque le constructeur renvoie un objet, l'objet est renvoyé directement. 🎜🎜🎜Simulation de la version finale🎜rrreee🎜[Recommandations associées : 🎜Tutoriel vidéo javascript🎜]🎜

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:csdn.net
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal