7个简单但棘手的JavaScript面试问题

开发 前端
在这篇文章中,你会发现7个乍一看很简单但很棘手的JavaScript面试问题。

[[318982]]

如果您符合高级开发人员的资格,其工作涉及JavaScript,那么在编码面试中很有可能会被问到棘手的问题。

遵循这个建议:“熟能生巧”。深入有规律地学习JavaScript将提高您的编码能力,并且可以提高您的面试技巧。

在这篇文章中,你会发现7个乍一看很简单但很棘手的JavaScript面试问题。

虽然一开始这些问题看起来是随意的,但是它们试图与JavaScript的重要概念挂钩。所以你最好在下次面试前练习一下!

1. 意外全局变量

Question

以下代码段中用于运算 typeof a 和 typeof b 的内容: 

  1. function foo() {  
  2.   let a = b = 0;  
  3.   a++;  
  4.   return a;  
  5.  
  6. foo();  
  7. typeof a; // => ???  
  8. typeof b; // => ??? 

Answer

让我们看一下第2行:let a = b = 0,该语句声明一个局部变量a,然儿,它也声明了全局变量b。

在 foo() 范围或全局范围中都没有声明变量b。因此JavaScript将 b = 0 表达式解释为 window.b = 0。换句话说,b是意外创建的全局变量。

在浏览器中,以上代码片段等效于: 

  1. function foo() {  
  2.   let a;  
  3.   window.b = 0 
  4.   a = window.b;  
  5.   a++;  
  6.   return a;  
  7.  
  8. foo();  
  9. typeof a;        // => 'undefined'  
  10. typeof window.b; // => 'number' 

typeof a 等于 'undefined',变量 a 存在于 foo() 范围内,而在外部范围内不使用。

因为b是一个值为0的全局变量,所以b的类型的值为 'number'。

2.数组的length属性

Question

clothes[0] 的值是什么? 

  1. const clothes = ['jacket', 't-shirt'];  
  2. clothes.length = 0 
  3. clothes[0]; // => ??? 

Answer

数组对象的 length 属性具有特殊的行为:减少length属性的值的副作用是删除自己的数组元素。因此,当JavaScript执行 clothes.length = 0 的时候将删除所有元素。

clothes [0] 等于 undefined 的,因为 clothes 数组已被清空。

3.鹰眼测试

Question

numbers 数组的内容是什么? 

  1. const length = 4 
  2. const numbers = [];  
  3. for (var i = 0; i < length; i++);{  
  4.   numbers.push(i + 1);  
  5.  
  6. numbers; // => ??? 

让我们仔细看一下分号 ; 出现在左大括号 {:

这个分号很容易被忽略,它创建了一个空语句。空语句是不做任何事情的空语句。

for() 在空语句上进行4次迭代(不执行任何操作),而忽略实际将项目推入数组的块:{number.push(i + 1);}。上面的代码等效于以下代码: 

  1. const length = 4 
  2. const numbers = [];  
  3. var i;  
  4. for (i = 0; i < length; i++) {  
  5.   // does nothing  
  6.  
  7. {   
  8.   // a simple block  
  9.   numbers.push(i + 1);  
  10.  
  11. numbers; // => [5] 

for() 将 i 变量递增到4,然后JavaScript一次进入块 {number.push(i + 1);},将 4 +1 推入数字数组。

因此,numbers 数组的内容为 [5]。

4.自动分号插入

Question 

  1. arrayFromValue() 返回什么值?  
  2. function arrayFromValue(item) {  
  3.   return  
  4.     [item];  
  5.  
  6. arrayFromValue(10); // => ??? 

Answer

很容易错过 return 关键字和 [item] 表达式之间的换行符。此换行符使JavaScript自动在 return 和 [item] 表达式之间插入分号。

这是等效的代码,在返回后插入了分号: 

  1. function arrayFromValue(item) {  
  2.   return;  
  3.   [item];  
  4.  
  5. arrayFromValue(10); // => undefined 

return; 函数内部使其返回 undefined。因此 arrayFromValue(10) 的值为 undefined。

5.经典问题:棘手的闭包

Question

以下代码将输出什么到控制台? 

  1. let i;  
  2. for (i = 0; i < 3; i++) {  
  3.   const log = () => {  
  4.     console.log(i);  
  5.   }  
  6.   setTimeout(log, 100);  

Answer

如果您以前从未听说过这个棘手的问题,则很可能您的答案是 0、1 和 2:这是错误的。当我第一次尝试解决它时,这也是我的答案!

执行此代码段有两个阶段。

Phase 1

  1.  for() 重复3次,在每次迭代过程中,都会创建一个新的函数 log() 来捕获变量 i。然后 setTimout() 计划执行log()。
  2.  当 for() 循环完成时,i 变量的值为 3。

log() 是一个捕获变量 i 的闭包,该变量在 for() 循环的外部范围中定义。请务必注意,闭包可以词法捕获 i 变量。

Phase 2

第二阶段发生在100ms之后:setTimeout() 调用了3次计划的 log() 。log() 读取变量i的当前值为3,并记录到控制台3。这就是为什么控制台输出为3、3 和 3 的原因。

如果您难以理解闭包,建议阅读“ JavaScript闭包的简单说明”。

您知道如何将代码段记录为0、1和2吗?请在下面的评论中写下您的解决方案!

6.浮点数计算

Question

这个等式的结果是什么? 

  1. 0.1 + 0.2 === 0.3 // => ??? 

Answer

首先,让我们看一下 0.1 + 0.2 的值: 

  1. 0.1 + 0.2; // => 0.30000000000000004 

0.1 和 0.2 的总和不完全是 0.3,而是略高于 0.3。

由于以二进制方式对浮点数进行编码,因此像浮点数相加之类的操作会产生舍入误差。

简而言之,直接比较浮点数并不精确。

因此 0.1 + 0.2 === 0.3 的结果是 false。

点击 0.30000000000000004.com 了解更多信息。

7.变量提升

Question

如果在声明前访问 myVar 和 myConst,会发生什么情况? 

  1. myVar;   // => ???  
  2. myConst; // => ???  
  3. var myVar = 'value' 
  4. const myConst = 3.14; 

Answer

提升和临时死区是影响JavaScript变量生命周期的两个重要概念。

在声明之前访问 myVar 的结果为 undefined。在初始化之前,提升的 var 变量具有 undefined 的值。

然儿,在声明行之前访问 myConst 会引发 ReferenceError。const 变量处于临时死区,直到声明行 const myConst = 3.14。

8.最后...

您可以认为某些问题对面试毫无用处。我有同样的感觉,特别是关于鹰眼测试。尽管如此,他们可能会被问到。

无论如何,其中许多问题都可以真正评估您是否精通JavaScript,例如棘手的闭包。如果您在阅读帖子时遇到一些困难,这很好地表明了您接下来必须学习什么! 

 

责任编辑:庞桂玉 来源: segmentfault
相关推荐

2023-09-13 08:37:56

程序员面试catch

2017-09-15 10:36:17

前端JavaScript面试题

2013-12-03 10:11:41

LinuxLinux面试题

2023-09-20 08:03:32

JavaScript编程语言

2020-03-23 09:56:50

软件测试面试技术

2024-03-27 08:36:48

JavaScriptWeb开发前端开发

2019-12-17 18:11:09

跳槽那些事儿机器学习人工智能

2023-06-27 00:04:10

程序员JavaScript

2015-05-11 14:02:21

JavaJava集合面试问题答案

2018-08-21 13:25:01

编程语言Java面试题

2019-07-19 08:10:47

JavaScript代码语言

2018-12-10 11:30:06

Openstack面试命令

2014-02-27 15:42:08

Linux面试

2013-05-22 10:04:18

Hibernate面试问题

2020-01-14 10:17:13

深度学习人生第一份工作人工智能

2009-03-03 09:33:13

面试ORACLE

2013-09-30 09:08:30

面试创业

2021-02-10 07:38:43

Node.js后端框架

2020-04-16 09:54:39

JavaScript前端技术

2014-07-25 17:21:43

Linux Shell面试
点赞
收藏

51CTO技术栈公众号