七个简单但棘手的 JS 面试问题

智数星罗盘
• 阅读 3982
作者:Dmitri Pavlutin

翻译:疯狂的技术咋

原文:https://dmitripavlutin.com/si...

未经允许严禁转载

在软件开发中,我认为最需要注意的是:

  1. 编码面试
  2. 有毒的主管或猪队友

不是 JavaScript,this,CSS,IE 浏览器,而是上述两点!

如果你参加 JavaScript 高级开发面试,那么很有可能在编码面试中被问到一些棘手的问题。

我知道这是不公平的。一些不知名的人将你放在一边来审视你。这并不是令人愉快的经历。

你能做什么?

请遵循以下建议:“通过实践变得完美”。通过投入足够的时间(最好地定期进行)来深入了解 JavaScript,可以改善你的编码,并且作为积极的结果,可以改善面试技巧。

在本文中,你将发现 7 个简单而又棘手的 JavaScript 面试问题。

尽管这些问题看上去似乎比较随意,但它们涉及了 JavaScript 的重要概念。所以你最好在下次面试之前进行练习!

1. 意外的全局变量

问题

在以下代码段中, typeof atypeof b 的评估结果是什么:

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

foo();
typeof a; // => ???
typeof b; // => ???

答案

让我们仔细看看第2行:let a = b = 0。该语句声明了局部变量 a。但同时它也声明了 全局 变量 b

foo() 或全局作用域中都没有声明变量 b。因此,JavaScript 将 b = 0 表达式解释为 window.b = 0

七个简单但棘手的 JS 面试问题

b 是意外创建的全局变量。

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

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

clothes[0]; // => ???

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

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

2. 数组长度属性

问题

clothes[0] 的值是什么:

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

clothes[0]; // => ???

答案

数组对象的 length 属性具有特殊行为

🏛减小 length 属性的值有删除当前数组元素(索引值在新旧长度值之间)的副作用。

由于 length 的这种行为,当 JavaScript 执行 clothes.length = 0 时,会删除 clothes 数组中的所有项。

clothes[0]undefined, ,因为 clothes 数组已清空。

3.鹰眼测试

问题

numbers 数组的内容是什么:

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

numbers; // => ???

答案

仔细看一下在花括号 { 前出现的分号;

七个简单但棘手的 JS 面试问题

在创建 null 语句 时,很容易忽略这个分号。null 语句是不执行任何操作的空语句。

for() 在 null 语句上迭代4次(不执行任何操作),而忽略将元素实际存入数组的代码块:{numbers.push(i + 1); }

前面的代码等效于以下代码:

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

numbers; // => [5]

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

因此,numbers[5]

这个问题背后的故事

很久以前,当我面试第一份工作时,有人问了我这个问题。

在这次面试中,我在 1 小时内回答了 20 个编码问题。空语句问题也在其中。

当时急于解决问题,我没有看到在大括号 { 之前的分号;。所以我得出了错误的答案 [1、2、3、4]

由于这种不公平的把戏,我有些失望。我问面试官,这种诡计背后的原因是什么?面试官回答:

“因为我们需要高度重视细节的人。”

幸运的是,我最终并没有为那家公司工作。

究竟怎么看待这件事,我将结论留给你自己做出。

4.自动插入分号

问题

arrayFromValue() 返回什么值?

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

arrayFromValue(10); // => ???

答案

很容易忽略关键字 return 和表达式 [items] 之间的换行。

换行符会使 JavaScript 自动在 return[items] 表达式之间插入分号。

这是等效代码,在 return 之后插入了分号:

function arrayFromValue(item) {
  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);
}

答案

如果你以前没有听说过这个棘手的问题,则你的答案很可能是 012,这是错误的。当我第一次试着解决它时,这也是我的答案!

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

阶段1

  1. for() 迭代 3 次。在每次迭代时,都会创建一个新函数 log(),该函数将捕获变量 i。然后,setTimout() 调度 log() 的执行。
  2. for() 循环完成时,变量 i 的值为 3

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

阶段2

第二阶段发生在 100ms 之后:

  1. 3 个固定的 log() 回调由 setTimeout() 调用。 log() 读取变量 i当前值 3,并把 3 log 到控制台。

这就是为什么控制台输出为 333 的原因。

你知道如何将代码段修复为输出 0、1 和 3 吗?请在下面的评论中写下你的解决方案!

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

访问 0.30000000000000004.com 可以了解更多信息。

7. 提升

问题

如果在声明前访问 myVarmyConst,会发生什么?

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

var myVar = 'value';
const myConst = 3.14;

答案

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

七个简单但棘手的 JS 面试问题

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

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

请遵循详细介绍JavaScript变量提升一文中的指南,以更好地掌握提升。

8.关键要点

你可能会认为有些面试中的问题没什么用,我也有同样的感觉,特别是鹰眼测试。但是他们仍然可能会被问到。

无论如何,以上大多数问题都能真正评估你是否精通 JavaScript。如果你在阅读本文时难以回答某些问题,则意味着这些是你接下来必须要去学习的内容!

在面试中提出棘手的问题是否公平?请在评论中写下你的看法。


本文首发微信公众号:前端先锋

欢迎扫描二维码关注公众号,每天都给你推送新鲜的前端技术文章

七个简单但棘手的 JS 面试问题

欢迎继续阅读本专栏其它高赞文章:


点赞
收藏
评论区
推荐文章
blmius blmius
4年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(
Wesley13 Wesley13
4年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
1年前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Stella981 Stella981
4年前
AssemblyScript 入门指南[每日前端夜话0xEB]
每日前端夜话0xEB每日前端夜话,陪你聊前端。每天晚上18:00准时推送。正文共:2459 字预计阅读时间:10分钟作者:DannyGuo翻译:疯狂的技术宅来源:logrocket!(https://oscimg.oschina.net/oscnet/b880277c594152a503
Stella981 Stella981
4年前
Node.js 12中的ES模块[每日前端夜话0x9E]
每日前端夜话0x9E每日前端夜话,陪你聊前端。每天晚上18:00准时推送。正文共:2552字预计阅读时间:10 分钟作者:BrianDeSousa翻译:疯狂的技术宅来源:logrocket!(https://oscimg.oschina.net/oscnet/2ccaf94cecd3
Stella981 Stella981
4年前
Python基础教程,Python入门教程(非常详细)
<divclass"htmledit\_views"id"content\_views"<p<ahref"http://c.biancheng.net/python/base/"rel"nofollow"第1章Python编程基础</a</p<p1.<ahref"http://c.biancheng.net/view/
Wesley13 Wesley13
4年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
4年前
Dubbo爆出严重漏洞!可导致网站被控制、数据泄露!附解决方案
http://dy.163.com/v2/article/detail/F5FPIFRU0511Q1AF.html  !(http://dingyue.ws.126.net/2020/0216/125ec4c4p00q5rcrs0019d200ig009qg00ig009q.png)  来源:华为云  原文地址:https://w
Python进阶者 Python进阶者
2年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
智数星罗盘
智数星罗盘
Lv1
今夜月明人尽望,不知秋思落谁家。
文章
5
粉丝
0
获赞
0