7 个沙雕又带有陷阱的 JS 面试题

做者:Dmitri Pavlutinjavascript

译者:前端小智html

来源:dmitripavlutin前端


阿(a)里(li)云(yun)服务器很便宜火爆,今年比去年便宜,10.24~11.11购买是1年86元,3年229元,能够点我时行参与java


为了保证的可读性,本文采用意译而非直译。git

在 JS 面试中,常常会看到一些简单而又沙雕的题目,这些题目包含一些陷阱,但这些在咱们规范的编码下或者业务中基本不会出现。 有些面试官就是这样,不专一于制定代码的标准和规范上,却用不规范的代码去检验别人是否细心。github

这魔幻的世界就是一个攀比优越感的,我能考你,我就是比你优越,真实。面试

来看看这 7 个沙雕题目是哪些。数组

1. 偶然建立的全局变量

面试官问浏览器

在下面的代码中 typeof atypeof b 结果各自是什么?(沙雕)服务器

function foo() {
  let a = b = 0;
  a++;
  return a;
}

foo();
typeof a; // => ???
typeof b; // => ???
复制代码

答案

这个代码的重点在第二行:let a = b = 0。这个语句声明了一个局部变量 a,可是它也声明了一个全局变量b

foo() 做用域或全局做用域中都没有声明变量 b。所以 JS 引荐将b = 0 表达式解释为 window.b = 0

以下图所示,函数 foo 中的 i 都是一个偶然建立的全局变量:

一样,在我们的问题中,b 是一个偶然建立的全局变量。在浏览器中,上面的代码至关于以下:

function foo() {
  let a;
  window.b = 0;
  a = window.b;
  a++;
  return a;
}

foo();
typeof a;        // => 'undefined'
typeof window.b; // => 'number'
复制代码

typeof a'undefined'。变量 a 仅在 foo() 做用域中声明,在外部做用域内不可用。

typeof b 结果是 'number'b 是一个值为 0 的全局变量

2. 数组的 length 属性

面试官问

clothes[0] 的值是什么?(沙雕)

const clothes = ['jacket', 't-shirt'];
clothes.length = 0;

clothes[0]; // => ???
复制代码

答案

数组对象的 length 属性具备一些特殊的行为

减小 length 属性的值的反作用是删除 本身的 数组元素,这些元素的数组索引位于新旧长度值之间。

因为 length 属性行为,当 JS 执行 clothes.length = 0 时,删除全部的 clothes 项。 因此 clothes[0] 的值为 undefined,由于 clothes 数组已被清空。

3.考验眼力的魔幻题

面试官问

下面代码中 numbers 数组的内容是什么? 注意 for() 后加了一个分号(;),真是沙雕

const length = 4;
const numbers = [];
for (var i = 0; i < length; i++);{
  numbers.push(i + 1);
}

numbers; // => ???
复制代码

答案

上面代码中 for() 后加了一个分号(;) ,加上分号,JS 会认为该语句结束,因此 for 循环执行了4次空语句,当退出循环的时候,此时的 i 值为 4。

而后执行 { numbers.push(i + 1); },因此最终 numbers 内容只有一个数字 5

上面的代码至关于下面的代码

const length = 4;
const numbers = [];
var i;
for (i = 0; i < length; i++) {
  // does nothing
}
{ 
  // a simple block
  numbers.push(i + 1);
}

numbers; // => [5]    
复制代码

用不规范的代码去检验别人是否细心,我以为很沙雕。

4.自动分号插入

面试官问

arrayFromValue() 返回什么值?(沙雕)

function arrayFromValue(items) {
  return
    [items];
}

arrayFromValue(10); // => ???
复制代码

答案

这里须要注意的 return[items] 之间已经换行了,JS 会在换行之间自动插入分号。因此上面等价下面的代码:

function arrayFromValue(items) {
  return;
  [items];
}

arrayFromValue(10); // => undefined
复制代码

return;在函数内部使该函数返回 undefined,因此 arrayFromValue(10) 的值为 undefined

5. 被考烂的一个经典闭包问题

面试官问

下面的代码执行结果是什么?(能不能换个题)

let i;
for (i = 0; i < 3; i++) {
  const log = () => {
    console.log(i);
  }
  setTimeout(log, 100);
}
复制代码

答案

当你对 JS 基础不是很了解的时候,很容易给出 0, 1, 2 的答案,我第一次在学校遇到这个题目也是这个答案。

执行这段代码的过程有两个阶段。

阶段1

  1. for() 迭代 3 次。在每次迭代时,都会建立一个新函数 log(),该函数将捕获变量 i。而后,setTimout() 调度 log() 的执行。

  2. for() 循环完成时,变量 i 的值为 3

log() 是一个捕获变量 i 的闭包,该变量在 for() 循环的外部做用域中定义。重要的是要了解闭包在词法上捕获了变量 i

阶段 2

第二阶段发生在 100 毫秒以后

setTimeout()调用 3log() 回调。log() 读取变量 i 的当前值,即 3

这就是为何控制台输出为 333 的缘由

6. 浮点运算

面试官问

下面的代码输出是什么? (能不能换个题)

0.1 + 0.2 === 0.3 // => ???
复制代码

答案

首先,来看一下 0.1 + 0.2 的值

0.1 + 0.2; // => 0.30000000000000004
复制代码

0.10.2 的和不等于 0.3,但略高于 0.3

因为以二进制方式对浮点数进行编码,所以像浮点数相加之类的操做会产生舍入偏差。

所以, 0.1 + 0.2 === 0.3false

7. 变量的提高

面试官问

若是在声明以前访问 myVarmyConst 会发生什么?(能不能换个题)

myVar;   // => ???
myConst; // => ???

var myVar = 'value';
const myConst = 3.14;    
复制代码

答案

提高和时间死区是影响 JS 变量生命周期的两个重要概念。

在声明以前访问 myVar 的结果是 undefined,由于使用 var 声明的变量会被提高且值为 undefined

可是,在声明行以前访问 myConst 会引起 ReferenceError。在代码行 const myConst = 3.14 以前,const 变量处于临时死区。

总结

你有没有感受上面的问题,有些是对面试毫无用处,特别是第3道题目。可是,仍是有一小部分的面试中会被问到。

固然,虽然都说是沙雕题目,但这些是能够评估你是否精通 JS,仍是有咱们学习的知识的。

在面试中,你还遇到哪些像这样的题目,欢迎留言讨论。

代码部署后可能存在的BUG无法实时知道,过后为了解决这些BUG,花了大量的时间进行log 调试,这边顺便给你们推荐一个好用的BUG监控工具 Fundebug

原文:dmitripavlutin.com/simple-but-…

交流(欢迎加入群,群工做日都会发红包,互动讨论技术)

阿里云最近在作活动,低至2折,有兴趣能够看看:promotion.aliyun.com/ntms/yunpar…

干货系列文章汇总以下,以为不错点个Star,欢迎 加群 互相学习。

github.com/qq449245884…

由于篇幅的限制,今天的分享只到这里。若是你们想了解更多的内容的话,能够去扫一扫每篇文章最下面的二维码,而后关注我们的微信公众号,了解更多的资讯和有价值的内容。

clipboard.png

每次整理文章,通常都到2点才睡觉,一周4次左右,挺苦的,还望支持,给点鼓励

相关文章
相关标签/搜索