深入浅出:从JS的new运算符到手写ES5/ES6版实现

深入浅出:从JS的new运算符到手写ES5/ES6版实现

一、JS中的new运算符本质

当我们使用new Person('awei',20)时,JS引擎会默默完成以下关键步骤:

javascript 复制代码
function Person(name, age) {
  this.name = name
  this.age = age
}
Person.prototype.sayName = function() {
  console.log(this.name)
}

const awei = new Person('awei', 20)

执行流程解析

  1. 创建纯净对象 :在堆内存中开辟新空间,生成空对象{}

  2. 建立原型链 :将空对象的__proto__指向构造函数的prototype

  3. 绑定执行上下文 :将构造函数内部的this指向这个新对象

  4. 执行构造函数 :相当于执行Person.call(obj, 'awei', 20)

  5. 返回结果处理

    • 构造函数无返回值时自动返回新对象
    • 若返回非对象类型则仍返回新对象
    • 若返回对象则直接使用该返回值

二、ES5环境手写new实现

ini 复制代码
function objectFactory() {
  const obj = new Object();
  const Constructor = [].shift.call(arguments);
  
  obj.__proto__ = Constructor.prototype;
  const ret = Constructor.apply(obj, arguments);
  
  return typeof ret === 'object' ? ret : obj;
}

// 测试用例
let awei = objectFactory(Person, 'awei', 20)

实现要点

  1. 通过arguments处理参数(类数组转数组)

  2. 使用shift方法分离构造函数

  3. 显式设置原型链(比JSON创建对象更灵活)

  4. 处理构造函数返回值:

    • 返回对象时优先采用
    • 非对象类型返回新实例

三、ES6环境优化版实现

javascript 复制代码
function objectFactory(fn, ...args) {
  const obj = {};
  obj.__proto__ = fn.prototype;
  
  const ret = fn.apply(obj, args);
  return ret instanceof Object ? ret : obj;
}

// 支持特殊返回值
function Person() {
  return { tag: 'custom object' }
}

const instance = objectFactory(Person)

升级优化点

  1. 参数处理:使用剩余参数...args替代arguments
  2. 空对象创建:对象字面量{}更高效
  3. 返回值判断:instanceof检测更严谨
  4. 支持箭头函数:通过原型链动态绑定

四、核心差异对比

特性 ES5实现 ES6实现
参数处理 arguments+shift 剩余参数...args
空对象创建 new Object() 对象字面量{}
原型设置 显式__proto__赋值 同左
返回值判断 typeof检测 instanceof检测
构造函数类型支持 仅标准函数 支持箭头函数(需特殊处理)

特殊场景处理

  • 当构造函数返回null时,typeof null === 'object'null instanceof Object === false
  • 箭头函数作为构造函数时需配合Object.setPrototypeOf处理
  • 需要兼容SymbolBigInt等基本类型返回值时需扩展类型判断

理解new运算符的底层实现,不仅能帮助开发者应对面试考点,更能深入理解原型链工作机制。当我们需要创建具有特殊初始化逻辑的对象,或需要自定义对象创建过程时,手写new的实现方式将展现出强大的灵活性。

相关推荐
漆黑骑士1 天前
Web Component
前端
San301 天前
深入理解 JavaScript 事件机制:从事件流到事件委托
前端·javascript·ecmascript 6
行走在顶尖1 天前
基础随记
前端
Sakura_洁1 天前
解决 el-table 在 fixed 状态下获取 dom 不准确的问题
前端
best6661 天前
Vue3什么时候不会触发onMounted生命周期钩子?
前端·vue.js
best6661 天前
Javascript有哪些遍历数组的方法?哪些不支持中断?那些不支持异步遍历?
前端·javascript·面试
特级业务专家1 天前
Chrome DevTools 高级调试技巧:从入门到真香
前端·javascript·浏览器
爱学习的程序媛1 天前
【Web前端】Angular核心知识点梳理
前端·javascript·typescript·angular.js
小时前端1 天前
前端架构师视角:如何设计一个“站稳多端”的跨端体系?
前端·javascript·面试
p***h6431 天前
JavaScript图像处理开发
开发语言·javascript·图像处理