深入浅出:从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的实现方式将展现出强大的灵活性。

相关推荐
我这里是好的呀几秒前
全栈开发个人博客13.AI聊天设计
前端·全栈
金金金__1 分钟前
Element-Plus:popconfirm与tooltip一起使用不生效?
前端·vue.js·element
lyc2333332 分钟前
小L带你看鸿蒙应用升级的数据迁移适配📱
前端
用户26812851066698 分钟前
react-pdf(pdfjs-dist)如何兼容老浏览器(chrome 49)
前端
阿怼丶8 分钟前
🚀 如何在内网中运行 Cesium?基于 NestJS 构建离线地形与影像服务
前端·gis
lyc2333339 分钟前
鸿蒙应用升级场景下的数据迁移适配
前端
DuxWeb10 分钟前
深入 Vue3 的类型传递机制与 React 的区别
前端
前端梭哈攻城狮12 分钟前
uniapp图片上传添加水印/压缩/剪裁
前端·javascript·vue.js
天涯学馆12 分钟前
前后端分离的 API 设计:技术深度剖析
前端·javascript·面试
lyc23333313 分钟前
鸿蒙File Access Framework:用户文件访问与权限控制📂
前端