做者:Dmitri Pavlutin
译者:前端小智
来源:dmitripavlutin
这几天本身的公众号无套路送现金 200+,参与方式以下
https://mp.weixin.qq.com/s/PT...
为了保证的可读性,本文采用意译而非直译。javascript
在 JS 面试中,常常会看到一些简单而又沙雕的题目,这些题目包含一些陷阱,但这些在咱们规范的编码下或者业务中基本不会出现。 有些面试官就是这样,不专一于制定代码的标准和规范上,却用不规范的代码去检验别人是否细心。html
这魔幻的世界就是一个攀比优越感的,我能考你,我就是比你优越,真实。前端
来看看这 7 个沙雕题目是哪些。java
面试官问git
在下面的代码中 typeof a
和 typeof b
结果各自是什么?(沙雕)github
function foo() { let a = b = 0; a++; return a; } foo(); typeof a; // => ??? typeof b; // => ???
答案面试
这个代码的重点在第二行
:let a = b = 0
。这个语句声明了一个局部变量 a
,可是它也声明了一个全局变量b
。segmentfault
在 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
的全局变量
面试官问
clothes[0]
的值是什么?(沙雕)
const clothes = ['jacket', 't-shirt']; clothes.length = 0; clothes[0]; // => ???
答案
数组对象的 length
属性具备一些特殊的行为:
减小length
属性的值的反作用是删除本身的
数组元素,这些元素的数组索引位于新旧长度值之间。
因为 length
属性行为,当 JS 执行 clothes.length = 0
时,删除全部的 clothes
项。 因此 clothes[0]
的值为 undefined
,由于 clothes
数组已被清空。
面试官问
下面代码中 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]
用不规范的代码去检验别人是否细心,我以为很沙雕。
面试官问
arrayFromValue()
返回什么值?(沙雕)
function arrayFromValue(item) { return [items]; } arrayFromValue(10); // => ???
答案
这里须要注意的 return
和 [items]
之间已经换行了,JS 会在换行之间自动插入分号。因此上面等价下面的代码:
function arrayFromValue(item) { return; [items]; } arrayFromValue(10); // => undefined
return;
在函数内部使该函数返回 undefined
,因此 arrayFromValue(10)
的值为 undefined
。
面试官问
下面的代码执行结果是什么?(能不能换个题)
let i; for (i = 0; i < 3; i++) { const log = () => { console.log(i); } setTimeout(log, 100); }
答案
当你对 JS 基础不是很了解的时候,很容易给出 0, 1, 2
的答案,我第一次在学校遇到这个题目也是这个答案。
执行这段代码的过程有两个阶段。
阶段1
for()
迭代 3
次。在每次迭代时,都会建立一个新函数 log()
,该函数将捕获变量 i
。而后,setTimout()
调度 log()
的执行。for()
循环完成时,变量 i
的值为 3
。log()
是一个捕获变量 i
的闭包,该变量在 for()
循环的外部做用域中定义。重要的是要了解闭包在词法上捕获了变量 i
。
阶段 2
第二阶段发生在 100
毫秒以后
setTimeout()调用 3
个 log()
回调。log()
读取变量 i
的当前值,即 3
。
这就是为何控制台输出为 3
, 3
和 3
的缘由
其实还有一种解释,就是计算机执行一条语句基本都是纳秒级别的,因此执行一个简单循环根本花费不了多少,当 100
毫秒后,for
早就跑完了,变量 i
已是退出循环的条件值。
面试官问
下面的代码输出是什么? (能不能换个题)
0.1 + 0.2 === 0.3 // => ???
答案
首先,来看一下 0.1 + 0.2
的值
0.1 + 0.2; // => 0.30000000000000004
0.1
和 0.2
的和不等于 0.3
,但略高于 0.3
。
因为以二进制方式对浮点数进行编码,所以像浮点数相加之类的操做会产生舍入偏差。
所以, 0.1 + 0.2 === 0.3
是 false
。
面试官问
若是在声明以前访问 myVar
和 myConst
会发生什么?(能不能换个题)
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。
原文:https://dmitripavlutin.com/si...
阿里云最近在作活动,低至2折,有兴趣能够看看:https://promotion.aliyun.com/...
干货系列文章汇总以下,以为不错点个Star,欢迎 加群 互相学习。
https://github.com/qq44924588...
由于篇幅的限制,今天的分享只到这里。若是你们想了解更多的内容的话,能够去扫一扫每篇文章最下面的二维码,而后关注我们的微信公众号,了解更多的资讯和有价值的内容。
每次整理文章,通常都到2点才睡觉,一周4次左右,挺苦的,还望支持,给点鼓励