关于javascript:详解-JavaScript-构造函数和-new-操作符

4次阅读

共计 3711 个字符,预计需要花费 10 分钟才能阅读完成。

结构器和操作符 “new”

惯例的 {...} 语法容许创立一个对象。然而咱们常常须要创立许多相似的对象,例如多个用户或菜单项等。

这能够应用构造函数和 "new" 操作符来实现。

构造函数

构造函数在技术上是惯例函数。不过有两个约定:

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

例如:

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 来运行。

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:prototypes> 和 <info: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


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

正文完
 0