详解 JavaScript 构造函数和 "new" 操做符

构造器和操做符 "new"

常规的 {...} 语法容许建立一个对象。可是咱们常常须要建立许多相似的对象,例如多个用户或菜单项等。javascript

这可使用构造函数和 "new" 操做符来实现。html

构造函数

构造函数在技术上是常规函数。不过有两个约定:java

  1. 它们的命名以大写字母开头。
  2. 它们只能由 "new" 操做符来执行。

例如:react

function User(name) {
  this.name = name;
  this.isAdmin = false;
}

let user = new User("Jack");

alert(user.name); // Jack
alert(user.isAdmin); // false

当一个函数被使用 new 操做符执行时,它按照如下步骤:算法

  1. 一个新的空对象被建立并分配给 this
  2. 函数体执行。一般它会修改 this,为其添加新的属性。
  3. 返回 this 的值。

换句话说,new User(...) 作的就是相似的事情:微信

function User(name) {
  // this = {};(隐式建立)

  // 添加属性到 this
  this.name = name;
  this.isAdmin = false;

  // return this;(隐式返回)
}

因此 new User("Jack") 的结果是相同的对象:函数

let user = {
  name: "Jack",
  isAdmin: false
};

如今,若是咱们想建立其余用户,咱们能够调用 new User("Ann")new User("Alice") 等。比每次都使用字面量建立要短得多,并且更易于阅读。学习

这是构造器的主要目的 —— 实现可重用的对象建立代码。测试

让咱们再强调一遍 —— 从技术上讲,任何函数均可以用做构造器。即:任何函数均可以经过 new 来运行,它会执行上面的算法。“首字母大写”是一个共同的约定,以明确表示一个函数将被使用 new 来运行。this

new function() { ... }

若是咱们有许多行用于建立单个复杂对象的代码,咱们能够将它们封装在构造函数中,像这样:

let user = new function() {
  this.name = "John";
  this.isAdmin = false;

  // ……用于用户建立的其余代码
  // 也许是复杂的逻辑和语句
  // 局部变量等
};

构造器不能被再次调用,由于它不保存在任何地方,只是被建立和调用。所以,这个技巧旨在封装构建单个对象的代码,而无需未来重用。

构造器模式测试:new.target

进阶内容:

本节涉及的语法内容不多使用,除非你想了解全部内容,不然你能够直接跳过该语法。

在一个函数内部,咱们可使用 new.target 属性来检查它是否被使用 new 进行调用了。

对于常规调用,它为空,对于使用 new 的调用,则等于该函数:

function User() {
  alert(new.target);
}

// 不带 "new":
User(); // undefined

// 带 "new":
new User(); // function User { ... }

它能够被用在函数内部,来判断该函数是被经过 new 调用的“构造器模式”,仍是没被经过 new 调用的“常规模式”。

咱们也可让 new 调用和常规调用作相同的工做,像这样:

function User(name) {
  if (!new.target) { // 若是你没有经过 new 运行我
    return new User(name); // ……我会给你添加 new
  }

  this.name = name;
}

let john = User("John"); // 将调用重定向到新用户
alert(john.name); // John

这种方法有时被用在库中以使语法更加灵活。这样人们在调用函数时,不管是否使用了 new,程序都能工做。

不过,处处都使用它并非一件好事,由于省略了 new 使得很难观察到代码中正在发生什么。而经过 new 咱们均可以知道这建立了一个新对象。

构造器的 return

一般,构造器没有 return 语句。它们的任务是将全部必要的东西写入 this,并自动转换为结果。

可是,若是这有一个 return 语句,那么规则就简单了:

  • 若是 return 返回的是一个对象,则返回这个对象,而不是 this
  • 若是 return 返回的是一个原始类型,则忽略。

换句话说,带有对象的 return 返回该对象,在全部其余状况下返回 this

例如,这里 return 经过返回一个对象覆盖 this

function BigUser() {

  this.name = "John";

  return { name: "Godzilla" };  // <-- 返回这个对象
}

alert( new BigUser().name );  // Godzilla,获得了那个对象

这里有一个 return 为空的例子(或者咱们能够在它以后放置一个原始类型,没有什么影响):

function SmallUser() {

  this.name = "John";

  return; // <-- 返回 this
}

alert( new SmallUser().name );  // John

一般构造器没有 return 语句。这里咱们主要为了完整性而说起返回对象的特殊行为。

省略括号

顺便说一下,若是没有参数,咱们能够省略 new 后的括号:

let user = new User; // <-- 没有参数
// 等同于
let user = new User();

这里省略括号不被认为是一种“好风格”,可是规范容许使用该语法。

构造器中的方法

使用构造函数来建立对象会带来很大的灵活性。构造函数可能有一些参数,这些参数定义了如何构造对象以及要放入什么。

固然,咱们不只能够将属性添加到 this 中,还能够添加方法。

例如,下面的 new User(name) 用给定的 name 和方法 sayHi 建立了一个对象:

function User(name) {
  this.name = name;

  this.sayHi = function() {
    alert( "My name is: " + this.name );
  };
}

let john = new User("John");

john.sayHi(); // My name is: John

/*
john = {
   name: "John",
   sayHi: function() { ... }
}
*/

是用于建立复杂对象的一个更高级的语法,咱们稍后会讲到。

总结

  • 构造函数,或简称构造器,就是常规函数,但你们对于构造器有个共同的约定,就是其命名首字母要大写。
  • 构造函数只能使用 new 来调用。这样的调用意味着在开始时建立了空的 this,并在最后返回填充了值的 this

咱们可使用构造函数来建立多个相似的对象。

JavaScript 为许多内置的对象提供了构造函数:好比日期 Date、集合 Set 以及其余咱们计划学习的内容。

对象,咱们还会回来哒!

在本章中,咱们只介绍了关于对象和构造器的基础知识。它们对于咱们在下一章中,学习更多关于数据类型和函数的相关知识很是重要。

在咱们学习了那些以后,咱们将回到对象,在 info:prototypesinfo:classes 章节中深刻介绍它们。

做业题

先本身作题目再看答案。

1. 两个函数 — 一个对象

重要程度:⭐️⭐️

是否能够建立像 new A()==new B() 这样的函数 AB

function A() { ... }
function B() { ... }

let a = new A;
let b = new B;

alert( a == b ); // true

若是能够,请提供一个它们的代码示例。

2. 建立 new Calculator

重要程度:⭐️⭐️⭐️⭐️⭐️

建立一个构造函数 Calculator,它建立的对象中有三个方法:

  • read() 使用 prompt 请求两个值并把它们记录在对象的属性中。
  • sum() 返回这些属性的总和。
  • mul() 返回这些属性的乘积。

例如:

let calculator = new Calculator();
calculator.read();

alert( "Sum=" + calculator.sum() );
alert( "Mul=" + calculator.mul() );

3. 建立 new Accumulator

重要程度:⭐️⭐️⭐️⭐️⭐️

建立一个构造函数 Accumulator(startingValue)

它建立的对象应该:

  • 将“当前 value”存储在属性 value 中。起始值被设置到构造器 startingValue 的参数。
  • read() 方法应该使用 prompt 来读取一个新的数字,并将其添加到 value 中。

换句话说,value 属性是全部用户输入值与初始值 startingValue 的总和。

下面是示例代码:

let accumulator = new Accumulator(1); // 初始值 1

accumulator.read(); // 添加用户输入的 value
accumulator.read(); // 添加用户输入的 value

alert(accumulator.value); // 显示这些值的总和

答案:

在微信公众号「技术漫谈」后台回复 1-4-5 获取做业答案。


现代 JavaScript 教程:开源的现代 JavaScript 从入门到进阶的优质教程。React 官方文档推荐,与 MDN 并列的 JavaScript 学习教程

在线免费阅读:https://zh.javascript.info


扫描下方二维码,关注微信公众号「技术漫谈」,订阅更多精彩内容。

相关文章
相关标签/搜索