面试热点: 你能模拟实现 new 操作符吗

简介: 面试热点: 你能模拟实现 new 操作符吗

前言


什么是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(); 
复制代码


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


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


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


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


MDN 中对 new 的描述: 使用 new 来构建函数,会执行如下四部操作:

  1. 创建一个空的简单 JavaScript 对象(即 {} );
  2. 为步骤1新创建的对象添加属性 __proto__ ,将该属性链接至构造函数的原型对象 ;
  3. 将步骤1新创建的对象作为 this 的上下文 ;
  4. 如果该函数没有返回对象,则返回 this


补充说明


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


function Thin_User(name, age) {}
const u1 = new Thin_user(...)
const u2 = createObject(Thin_User, ...a)
复制代码


初步模拟


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


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


__proto__和prototype ,可以看JavaScript之彻底理解原型与原型链call/apply ,可以看[JavaScript之手撕call、apply]juejin.cn/post/702094…


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


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;
}
复制代码


返回值处理


众所周知,函数是有返回值的,那构造函数如果有返回值,最终执行 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(); 
复制代码


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


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


返回值为对象


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();
复制代码


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


image.png

发现xiaobao对象的age发生了改变,而且增加了fat属性,正好与构造函数的返回值一样。


看完这两个例子,基本可以理清构造函数有返回值的情况:当构造函数返回值为对象时,直接返回这个对象。


终版模拟


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;
}
复制代码


往期精彩文章



后语


如果大家感觉此文对你有一些帮助,希望能点个赞,鼓励鼓励阿包,阿包会不断努力的。另外如果本文章有问题,或者对文章其中一部分不理解,都可以评论区回复我,我们来一起讨论,共同学习,一起进步!


如果感觉评论区说不明白,也可以添加我的微信或者 qq 详细交流,名字都是战场小包。



相关文章
|
3月前
|
Java
【Java基础面试三十二】、new String(“abc“) 是去了哪里,仅仅是在堆里面吗?
这篇文章解释了Java中使用`new String("abc")`时,JVM会将字符串直接量"abc"存入常量池,并在堆内存中创建一个新的String对象,该对象会指向常量池中的字符串直接量。
|
6月前
|
JavaScript 前端开发 Java
【面试题】new 一个对象时,js 做了什么?
【面试题】new 一个对象时,js 做了什么?
|
6月前
|
JavaScript 前端开发 编译器
面试官问我new Vue阶段做了什么?
面试官问我new Vue阶段做了什么?
|
3月前
|
Java
【Java基础面试二十八】、使用字符串时,new和““推荐使用哪种方式?
这篇文章讨论了在Java中使用字符串时,推荐使用双引号`""`直接量方式而不是使用`new`操作符,因为`new`会在常量池之外额外创建一个对象,导致更多的内存占用。
|
6月前
|
编译器 程序员 C语言
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题(下)
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题
42 0
|
6月前
|
编译器 C语言 C++
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题(中)
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题
50 0
|
6月前
|
存储 程序员 编译器
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题(上)
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题
46 0
|
6月前
|
C++
面试题:malloc和new的区别
面试题:malloc和new的区别
120 0
每天一道面试题之String str=“i“与 String str=new String(“i”)一样吗?
每天一道面试题之String str=“i“与 String str=new String(“i”)一样吗?
|
前端开发
前端经典面试题 | New操作符的原理
前端经典面试题 | New操作符的原理