当前位置: 首页 > news >正文

javascript中new操作符的工作原理

在 JavaScript 中,new 操作符用于创建对象的实例。它可以让我们通过构造函数创建一个新的对象,并初始化该对象的属性和方法。尽管 new 操作符的使用很常见,但它在背后实际进行了几个步骤。下面详细解释 new 操作符具体做了哪些事情。

new 操作符的执行过程

当我们使用 new 来调用一个构造函数时,JavaScript 会自动执行以下步骤:

  1. 创建一个新的空对象

    • 创建一个新的对象(我们可以称它为 newObj),这个对象继承了构造函数的 prototype 属性。
    • newObj.__proto__ = Constructor.prototype
  2. this 绑定到新创建的对象上

    • 构造函数中的 this 会被绑定到新创建的对象(即 newObj)上。
    • 这样,在构造函数中定义的属性和方法就会被添加到新对象上。
  3. 执行构造函数的代码

    • 构造函数会被执行,并且 this 现在指向 newObj。构造函数内部可以为新对象添加属性或方法。
  4. 返回新对象

    • 如果构造函数显式返回一个对象,那么返回这个对象。
    • 如果构造函数没有显式返回对象,则默认返回创建的 newObj

示例

通过一个简单的例子来说明 new 的工作原理:

function Person(name, age) {this.name = name;this.age = age;this.sayHello = function() {console.log('Hello, my name is ' + this.name);};
}const person1 = new Person('Alice', 25);
console.log(person1.name);  // 输出: Alice
console.log(person1.age);   // 输出: 25
person1.sayHello();         // 输出: Hello, my name is Alice

在这个例子中,我们使用 new 来创建 Person 对象的实例 person1,这个对象拥有 nameage 属性,并且可以调用 sayHello 方法。

new 操作符的执行细节

让我们逐步看一下当执行 new Person('Alice', 25) 时,发生了什么:

  1. 创建一个新的空对象

    • 一个空对象 newObj 被创建,它的 __proto__ 指向 Person.prototype
    newObj = Object.create(Person.prototype);
    
  2. this 绑定到新对象上

    • 调用构造函数 Person 时,this 被绑定为 newObj,并且构造函数内部的 this.namethis.age 被赋值。
    Person.call(newObj, 'Alice', 25);
    
  3. 执行构造函数的代码

    • 构造函数内部的代码运行,为 newObj 添加了 nameage 属性以及 sayHello 方法。
  4. 返回新对象

    • 因为构造函数没有显式返回对象,所以默认返回 newObj

最终结果是,一个拥有 nameage 属性的对象 person1 被创建,并且继承了 Person.prototype 的所有属性和方法。

new 操作符的步骤模拟

我们可以通过手动模拟 new 操作符来理解它的内部工作原理。下面是一个简化的 new 的实现:

function myNew(constructor, ...args) {// 1. 创建一个新的空对象const obj = Object.create(constructor.prototype);// 2. 将 this 绑定到新的对象并执行构造函数const result = constructor.apply(obj, args);// 3. 如果构造函数返回的是一个对象,则返回该对象,否则返回新创建的对象return result instanceof Object ? result : obj;
}function Person(name, age) {this.name = name;this.age = age;
}const person1 = myNew(Person, 'Alice', 25);
console.log(person1.name);  // 输出: Alice
console.log(person1.age);   // 输出: 25

在上面的代码中,myNew 函数模拟了 new 的基本行为,包括创建对象、绑定 this、执行构造函数和返回新对象。

new 操作符中的返回值

  • 如果构造函数显式返回一个对象,则 new 操作符返回该对象。
  • 如果构造函数没有返回对象,new 操作符将返回创建的对象。

例如:

function Person(name) {this.name = name;return { customObject: true };
}const person1 = new Person('Alice');
console.log(person1);  // 输出: { customObject: true }function Animal(name) {this.name = name;// 没有显式返回值
}const animal1 = new Animal('Tiger');
console.log(animal1);  // 输出: Animal { name: 'Tiger' }

Person 的例子中,构造函数返回了一个对象 { customObject: true },因此 new 操作符返回了这个对象,而不是默认的新对象。

Animal 的例子中,构造函数没有显式返回对象,new 操作符会返回 new Animal() 创建的实例对象。

new 的使用场景

new 操作符最常用于构造函数模式或类来创建对象的实例。常见场景包括:

  1. 创建类的实例

    • JavaScript 中的类定义使用 class 语法可以通过 new 来创建实例。
    class Person {constructor(name) {this.name = name;}
    }const person1 = new Person('Alice');
    console.log(person1.name); // 输出: Alice
    
  2. 使用构造函数创建对象

    • 除了 class 语法,传统的构造函数模式也是通过 new 来创建实例。
    function Car(model) {this.model = model;
    }const car1 = new Car('Toyota');
    console.log(car1.model); // 输出: Toyota
    
  3. 内置对象的实例化

    • JavaScript 的一些内置对象也需要通过 new 来创建实例,如 DateRegExpArray 等。
    const today = new Date();
    console.log(today); // 输出当前日期和时间
    

总结

new 操作符在 JavaScript 中用于创建对象的实例,它在背后执行了以下几步:

  1. 创建一个新的空对象并将其原型指向构造函数的 prototype
  2. 将构造函数的 this 绑定到新对象上。
  3. 执行构造函数的代码,初始化新对象的属性。
  4. 返回这个新对象(如果构造函数返回对象,则返回该对象)。

通过 new 操作符,我们可以轻松创建复杂对象,并且通过原型链继承共享方法和属性。

相关文章:

javascript中new操作符的工作原理

在 JavaScript 中,new 操作符用于创建对象的实例。它可以让我们通过构造函数创建一个新的对象,并初始化该对象的属性和方法。尽管 new 操作符的使用很常见,但它在背后实际进行了几个步骤。下面详细解释 new 操作符具体做了哪些事情。 new 操…...

基于springboot+vue 旅游网站的设计与实现

基于springbootvue 旅游网站的设计与实现 摘 要 互联网发展至今,无论是其理论还是技术都已经成熟,而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播,搭配信息管理工具可以很好地为人们提供服务。针对信息管理混乱&#xff0c…...

Ansible集群服务部署案例

案例描述 本案例共讲述了多个节点部署Elk集群日志分析系统,分别在三个节点使用ansible部署Kibana、Logstash以及Elasticsearch服务。 案例准备 1. 规划节点 IP 主机名 节点 192.168.100.25 ansible Ansible节点 192.168.100.35 node1 Elasticsearch/Kiba…...

探索AI编程新境界:aider库揭秘

文章目录 **探索AI编程新境界:aider库揭秘**背景:为何选择aider?简介:aider是什么?安装指南:如何安装aider?功能演示:aider的简单用法实战应用:aider在不同场景下的使用常…...

SQL Server 2012 ldf日志文接太大的截断和收缩日志处理

SQL Server 2012 ldf日志文接太大的截断和收缩日志处理操作 --- SQL Server 2012 ldf日志文接太大的截断和收缩日志处理 ----- 查看所有 database 列表及详情 select * from sys.databases;-- 切换到指定的操作数据库 use testdb;-- 查询当前数据库的日志文件ID和逻辑文件名 S…...

java日志门面之JCL和SLF4J

文章目录 前言一、JCL1、JCL简介2、快速入门3、 JCL原理 二、SLF4J1、SLF4J简介2、快速入门2.1、输出动态信息2.2、异常信息的处理 3、绑定日志的实现3.1、slf4j实现slf4j-simple和logback3.2、slf4j绑定适配器实现log4j 4、桥接旧的日志框架4.1、log4j日志重构为slf4jlogback的…...

Oracle DB运维常用的视图及数据字典

List item 本文介绍一些Oracle DB日常运维最常用到(使用频率很高)的视图及数据字典 用户有关的常用视图: 1、 查看当前用户的缺省表空间* SQL>select username,default_tablespace from user_users; 2、 查看当前用户的角色 SQL>sele…...

vue.config.js devServer中changeOrigin的作用

问题 vue开发时,为了解决前端跨域问题,通常在vue.config.js配置 devServer proxy devServer: {proxy:{/api: {target: http://b.com,changeOrigin: false},}, }官方文档http-proxy options对changeOrigin的解释 option.changeOrigin: true/false, Defa…...

基于Ubuntu 20.04 LTS上部署MicroK8s(最小生产的 Kubernetes)

目录 文章目录 目录简介Kubernetes简介MicroK8s简介Ubuntu系统MicroK8s的优势安装环境基本要求执行安装命令加入群组(使用非 root 用户访问)开启 dashboard 仪表盘查看服务名称查看仪表盘开放的端口打开浏览器检查状态打开你想要的服务(使用附加组件)开始使用 microk8s访问 Kub…...

Spring:项目中的统一异常处理和自定义异常

介绍异常的处理方式。在项目中,都会进行自定义异常,并且都是需要配合统一结果返回进行使用。 1.背景引入 (1)背景介绍 为什么要处理异常?如果不处理项目中的异常信息,前端访问我们后端就是显示访问失败的…...

有点快要跟不上时代的感觉

团队的群里面有一个同事突然问了下,下面的这个 JavaScript 如何进行优化 var startIndex (start undefined || start null) ? null : start[0].Value;看上面的代码就是典型的判断和返回的问题。 如果是要调试的话也不是做不出来,但可能要花点时间&a…...

【pytorch】pytorch入门4:神经网络的卷积层

文章目录 前言一、定义概念 缩写二、性质三、代码总结参考文献 前言 使用 B站小土堆课程的笔记 一、定义概念 缩写 卷积层是神经网络中用于突出特征来进行分类任务的层。 二、性质 卷积核例子:vgg16 model 三、代码 添加库 python代码块import os import …...

【机器学习】探索LSTM:深度学习领域的强大时间序列处理能力

目录 🍔 LSTM介绍 🍔 LSTM的内部结构图 2.1 LSTM结构分析 2.2 Bi-LSTM介绍 2.3 使用Pytorch构建LSTM模型 2.4 LSTM优缺点 🍔 小结 学习目标 🍀 了解LSTM内部结构及计算公式. 🍀 掌握Pytorch中LSTM工具的使用. &…...

QT学习笔记之文件操作

你千万不要跟任何人谈起任何事。你只要一谈起&#xff0c;就会想念起每一个人来。 在ui界面添加一个LineEdit(lEt)、QPushButton(btn)、QWidget widget.cpp #include "widget.h" #include "ui_widget.h" #include <QFile> #include <QFileDialo…...

Mybatis XML配置文件操作数据库

Mybaits在操作数据库时&#xff0c;可以有两种方式&#xff1b;第一种是使用注解的方式操作&#xff0c;另一种是使用XML配置文件的方式&#xff1a;一般而言&#xff0c;若没有特别的要求&#xff0c;则编写一些简单的SQL语句&#xff0c;可以直接使用注解的方式&#xff1b;编…...

Ansible-template模块动态生成特定文件

文章目录 一、Jinja2介绍什么是主要特性安装基本用法进阶特性总结 Jinja2与Ansible关系1. 模板引擎2. Ansible 的依赖3. 变量和模板4. 动态生成配置5. 社区和生态系统总结 二、Ansible如何使用Jinja2使用template模块Jinja2文件中使用判断和循环Jinja2文件中使用判断语法 Jinja…...

【Hadoop】【vim编辑器】【~/.bashrc 文件】如何编辑

1. 进入 vim 编辑器 在终端中输入以下命令&#xff1a; vim ~/.bashrc 2. 进入插入模式 打开文件后&#xff0c;你将处于普通模式。在普通模式下&#xff0c;你不能直接编辑文本。 要进入插入模式&#xff0c;请按下 i 键。这时&#xff0c;你应该会看到屏幕底部出现 -- 插…...

vs code自动报错

让vs code自动报错&#xff0c; 点击插件 → 搜索error lens → 点击install&#xff0c; 下载完后&#xff0c;编写的代码有问题就会自动报错了。 5、修改默认缩进字符 点击设置&#xff08;settings&#xff09; → 点击常用设置 → 修改字符缩进。...

详细分析Nginx中的proxy_pass 末尾斜杠

目录 前言1. 基本知识2. Demo 前言 对于Nginx的讲解&#xff0c;更多推荐阅读&#xff1a; Nginx配置静态网页访问&#xff08;图文界面&#xff09;Nginx将https重定向为http进行访问的配置&#xff08;附Demo&#xff09;Nginx从入门到精通&#xff08;全&#xff09;详细分…...

数据结构:双指针—移动0(OJ283)

给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 请注意 &#xff0c;必须在不复制数组的情况下原地对数组进行操作。 示例 1: 输入: nums [0,1,0,3,12] 输出: [1,3,12,0,0]示例 2: 输入: nums [0] 输出: […...

React Native 导航系统实战(React Navigation)

导航系统实战&#xff08;React Navigation&#xff09; React Navigation 是 React Native 应用中最常用的导航库之一&#xff0c;它提供了多种导航模式&#xff0c;如堆栈导航&#xff08;Stack Navigator&#xff09;、标签导航&#xff08;Tab Navigator&#xff09;和抽屉…...

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…...

基于服务器使用 apt 安装、配置 Nginx

&#x1f9fe; 一、查看可安装的 Nginx 版本 首先&#xff0c;你可以运行以下命令查看可用版本&#xff1a; apt-cache madison nginx-core输出示例&#xff1a; nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...

微信小程序 - 手机震动

一、界面 <button type"primary" bindtap"shortVibrate">短震动</button> <button type"primary" bindtap"longVibrate">长震动</button> 二、js逻辑代码 注&#xff1a;文档 https://developers.weixin.qq…...

用docker来安装部署freeswitch记录

今天刚才测试一个callcenter的项目&#xff0c;所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...

Mysql中select查询语句的执行过程

目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析&#xff08;Parser&#xff09; 2.4、执行sql 1. 预处理&#xff08;Preprocessor&#xff09; 2. 查询优化器&#xff08;Optimizer&#xff09; 3. 执行器…...

解读《网络安全法》最新修订,把握网络安全新趋势

《网络安全法》自2017年施行以来&#xff0c;在维护网络空间安全方面发挥了重要作用。但随着网络环境的日益复杂&#xff0c;网络攻击、数据泄露等事件频发&#xff0c;现行法律已难以完全适应新的风险挑战。 2025年3月28日&#xff0c;国家网信办会同相关部门起草了《网络安全…...

如何应对敏捷转型中的团队阻力

应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中&#xff0c;明确沟通敏捷转型目的尤为关键&#xff0c;团队成员只有清晰理解转型背后的原因和利益&#xff0c;才能降低对变化的…...

redis和redission的区别

Redis 和 Redisson 是两个密切相关但又本质不同的技术&#xff0c;它们扮演着完全不同的角色&#xff1a; Redis: 内存数据库/数据结构存储 本质&#xff1a; 它是一个开源的、高性能的、基于内存的 键值存储数据库。它也可以将数据持久化到磁盘。 核心功能&#xff1a; 提供丰…...

Python常用模块:time、os、shutil与flask初探

一、Flask初探 & PyCharm终端配置 目的: 快速搭建小型Web服务器以提供数据。 工具: 第三方Web框架 Flask (需 pip install flask 安装)。 安装 Flask: 建议: 使用 PyCharm 内置的 Terminal (模拟命令行) 进行安装,避免频繁切换。 PyCharm Terminal 配置建议: 打开 Py…...