JavaScript 是 所有现代浏览器 的官方语言。因此,各种语言的开发者面试中都会遇到 JavaScript 问题。

本文不讲最新的 JavaScript 库,通用开发实践,或任何新的 ES6 函数 。而是讲讲面试中经常出现的 3 个 JavaScript 问题。我问过这些问题,我的朋友说他们也问。

当然不是说你在准备 JavaScript 面试时只要学习这 3 个问题 —— 你还 有 很多 途径 去更好的准备即将到来的面试 —— 但面试官很有可能通过下面 3 个问题来判断你了解和掌握 JavaScript 和 DOM 的情况。

让我们开始吧!注意下面的例子中我们使用原生 JavaScript, 因为面试官通常想考查你在不借助库(例如 jQuery)的帮助时掌握 JavaScript 和 DOM 的情况。

问题 #1: 事件代理

创建应用时,有时需要给页面中的按钮,文字,或图片添加事件监听器,当用户与这些元素交互时触发某些操作。

我们以一个简单的代办事项列表为例,面试官会告诉你,他们希望在用户点击列表中某一项时触发一个动作。并让你用 JavaScript 根据下面的 HTML 代码实现这个功能:

  1. <ul id="todo-app">
  2. <li class="item">Walk the dog</li>
  3. <li class="item">Pay bills</li>
  4. <li class="item">Make dinner</li>
  5. <li class="item">Code for one hour</li>
  6. </ul>

你可能会像下面的代码一样给元素添加事件监听器:

  1. document.addEventListener('DOMContentLoaded', function() {
  2. let app = document.getElementById('todo-app');
  3. let items = app.getElementsByClassName('item');
  4. // 给每个列表项添加事件监听器
  5. for (let item of items) {
  6. item.addEventListener('click', function() {
  7. alert('you clicked on item: ' + item.innerHTML);
  8. });
  9. }
  10. });

当然上面的代码能完成面试官的需求,问题是每个列表项都会加上一个事件监听器。当列表只有 4 项时没有问题,但如果有人给代办事项列表新增了 10,000 个事项呢(他们也许有一大堆事情要做)?那时函数会创建 10,000 个事件监听器,然后把它们都添加到 DOM 上。这样 效率 非常低。

面试中最好首先问一下面试官用户最多可以添加多少个代办事项。如果永远不会超过 10 个,那上面的代码运行起来就没有问题。但如果用户输入待办事项的数量没有上限,那你就得换一个更高效的解决方案。

如果应用有上百个事件监听器,更高效的解决方案是给最外层的容器添加 一个 事件监听器,当用户真正点击的时候再去获取实际被点击的代办事项。这被称为 事件代理 ,这比给每个代办事项都单独添加事件监听器更高效。

下面是事件代理的代码:

  1. document.addEventListener('DOMContentLoaded', function() {
  2. let app = document.getElementById('todo-app');
  3. // 给容器添加事件监听器
  4. app.addEventListener('click', function(e) {
  5. if (e.target && e.target.nodeName === 'LI') {
  6. let item = e.target;
  7. alert('you clicked on item: ' + item.innerHTML);
  8. }
  9. });
  10. });

问题 #2: 在循环中使用闭包

面试中经常会问到闭包,因为面试官能通过这个问题的回答判断你对语言的熟悉程度,以及考察你是否知道什么时候使用闭包。

闭包就是能访问作用域外部变量的 内部函数 。闭包能用来实现 私有化 和创建 工厂函数 等作用。关于闭包的常见面试题是这样的:

写一个函数,循环一个整数数组,延迟 3 秒打印这个数组中每个元素的索引。

这个问题常见(不正确)的实现是这样:

  1. const arr = [10, 12, 15, 21];
  2. for (var i = 0; i < arr.length; i++) {
  3. setTimeout(function() {
  4. console.log('The index of this number is: ' + i);
  5. }, 3000);
  6. }

如果你运行这段函数,你会发现 3 秒之后每次都打印的是 4 ,而不是预期的 0, 1, 2, 3 。

为了正确的找到出现这种情况的原因,你需要理解 JavaScript 是如何运行这段代码的,这也是面试官想要考察你的地方。

原因是 setTimeout 函数创建了一个访问外部作用域的函数(闭包),就是包含索引 i 的那个循环。3 秒之后,函数开始执行打印 i 的值,而此时循环也结束了, i 的值已经是 4。因为循环遍历 0, 1, 2, 3, 4 后最终停在了 4。

实际上有 好几种方法 能 正确 解决这个问题。这里有两个:

  1. const arr = [10, 12, 15, 21];
  2. for (var i = 0; i < arr.length; i++) {
  3. // 给每个函数传入变量 i 让其能访问正确的索引
  4. setTimeout(function(i_local) {
  5. return function() {
  6. console.log('The index of this number is: ' + i_local);
  7. }
  8. }(i), 3000);
  9. }
  1. const arr = [10, 12, 15, 21];
  2. for (let i = 0; i < arr.length; i++) {
  3. // 使用 ES6 中的 let 关键字,它会在函数调用时创建一个新的绑定
  4. // 了解更多:http://exploringjs.com/es6/ch_variables.html#sec_let-const-loop-heads
  5. setTimeout(function() {
  6. console.log('The index of this number is: ' + i);
  7. }, 3000);
  8. }

问题 #3: Debouncing(防抖动)

有些浏览器事件能在很短的时间内被触发多次,例如调整窗口大小或滚动页面。如果你给窗口滚动事件添加一个事件监听器,然后用户不停地快速向下滚动页面,那你的事件可能在 3 秒之内被触发数千次。这会导致非常严重的性能问题。

如果在面试中讨论到构建应用程序,以及滚动事件,窗口调整事件,或者键盘事件等,请务必提及 debouncing 或者 throttling,作为提高页面速度与性能的方法。来一个 css-tricks 的实例:

2011 年,Twitter 出了一个问题:当滚动 Twitter 摘要时,页面变的很卡甚至无响应。John Resig 写了 一篇关于这个问题的博客 ,解释了直接将耗时的函数绑定在 scroll 事件上是一个多么糟糕的想法。

Debouncing 是解决这个问题的一种方法,它的做法是限制下次函数调用之前必须等待的时间间隔。正确实现 debouncing 的方法是将若干个函数调用 合成 一次,并在给定时间过去之后仅被调用一次。下面是一个原生 JavaScript 的实现,用到了 作用域 , 闭包, this , 和 计时事件 :

  1. // 将会包装事件的 debounce 函数
  2. function debounce(fn, delay) {
  3. // 维护一个 timer
  4. let timer = null;
  5. // 能访问 timer 的闭包
  6. return function() {
  7. // 通过 ‘this’ 和 ‘arguments’ 获取函数的作用域和变量
  8. let context = this;
  9. let args = arguments;
  10. // 如果事件被调用,清除 timer 然后重新设置 timer
  11. clearTimeout(timer);
  12. timer = setTimeout(function() {
  13. fn.apply(context, args);
  14. }, delay);
  15. }
  16. }

这个函数 — 当传入一个事件(fn)时 — 会在经过给定的时间(delay)后执行。

函数这样用:

  1. // 当用户滚动时被调用的函数
  2. function foo() {
  3. console.log('You are scrolling!');
  4. }
  5. // 在 debounce 中包装我们的函数,过 2 秒触发一次
  6. let elem = document.getElementById('container');
  7. elem.addEventListener('scroll', debounce(foo, 2000));

Throttling 是与 debouncing 类似的一种技术,但它不是在调用函数之前等待一段时间,throttling 是在较长的时间间隔内调用函数。所以如果一个事件每 100 毫秒被触发 10 次,throttling 会在每隔 2 秒时执行一次这个函数,而不是在 100 毫秒内执行 10 次事件。

面试中要注意的 3 个 JavaScript 问题的更多相关文章

  1. 面试中注意3个javascript的问题

    JavaScript 是所有现代浏览器的官方语言.因此,各种语言的开发者面试中都会遇到 JavaScript 问题. 本文不讲最新的 JavaScript 库,通用开发实践,或任何新的 ES6 函数. ...

  2. 前端js面试中的常见的算法问题

    虽说我们很多时候前端很少有机会接触到算法.大多都交互性的操作,然而从各大公司面试来看,算法依旧是考察的一方面.实际上学习数据结构与算法对于工程师去理解和分析问题都是有帮助的.如果将来当我们面对较为复杂 ...

  3. 解密面试中的套路,你都get到了么?

    如果大家有关注一些测试类的公众号或者论坛的话,肯定会发现很多文章都在表示现在行业的寒冬冷潮来了!然后有很多测试行业从业者,或者转行测试行业者都表示:工作好难找,公司跑了千千万,依然拿不到一个offer ...

  4. 在面试中忽然发现DateTime的一些...

    今天说说我面试中碰到的一个小问题,在我问起DateTime为什么无法赋值NULL值,一般第一反应都认为它是值类型,不是引用类型,但随后我查阅了度娘自我学习到它是结构类型,那么随之而然就无法赋值NULL ...

  5. 面试中关于Java你所需知道的的一切

    本篇文章会对面试中常遇到的Java技术点进行全面深入的总结,帮助我们在面试中更加得心应手,不参加面试的同学也能够借此机会梳理一下自己的知识体系,进行查漏补缺. 1. Java中的原始数据类型都有哪些, ...

  6. JAVA面试中问及HIBERNATE与 MYBATIS的对比,在这里做一下总结

    我是一名java开发人员,hibernate以及mybatis都有过学习,在java面试中也被提及问道过,在项目实践中也应用过,现在对hibernate和mybatis做一下对比,便于大家更好的理解和 ...

  7. 谈谈如何在面试中发掘程序猿的核心竞争力zz

    早两天看了知乎日报的这篇文章<什么是程序员的核心竞争力?>,caoz讲的几点是让我感同身受.这让我联想起了给程序猿的面试,其实也就是通过短暂的接触来发掘程序猿的核心竞争力.接下来我就谈谈我 ...

  8. java面试中的智力题

    智力题,每个正式的笔试.面试都会出,而且在面大企业的时候必然会问到,笔者曾在很多面试中,都被问到过,不过答得都不是很好,因为时间很短,加上我们有时候过于紧张,所以做出这类问题,还是有一定的难度,从这篇 ...

  9. 面试题_125_to_133_Java 面试中其他各式各样的问题

    这部分包含 Java 中关于 XML 的面试题,JDBC 面试题,正则表达式面试题,Java 错误和异常及序列化面试题 125)嵌套静态类与顶级类有什么区别?(答案)一个公共的顶级类的源文件名称与类名 ...

随机推荐

  1. oracle编程300例-性能优化(一)

    1.在SELECT语句中避免使用“*” 2.尽可能减小记录行数 3.使用rowid高效删除重复记录 实例: delete from stu s where s.rowid>(select min ...

  2. (Oracle)自定义调用AWR

    Oracle->自动发送AWR报告 2016年9月21日 09:31 需求描述: 每日或定期手动使用AWR报告来检查Oracle数据库状态不仅耗时也费力,需求使用脚本自动收集AWR报告.   分 ...

  3. ubuntu8.04下mysql更改用户和密码

    1.最近由于系统原因重装了mysql,但是发现安装过程中没有提示设置密码. 2.修改用户名和密码步骤 A.service mysql stop    #停止mysql服务 B.sudo vim /et ...

  4. JsonCpp在vs中使用

    Jsoncpp是c++生成和解析Json数据的跨平台开源库.下面简介如何在vs中使用. 1.官网下载.https://sourceforge.net/projects/jsoncpp/解压文件得到js ...

  5. python开发的学生管理系统

    python开发的学生管理系统(基础版) #定义一个函数,显示可以使用的功能列表给用户 def showInfo(): print("-"*30) print(" 学生管 ...

  6. hadoop生态搭建(3节点)-12.rabbitmq配置

    # 安装 需要相关包# ==================================================================node1 node2 node3 yum ...

  7. React 源码中的依赖注入方法

    一.前言 依赖注入(Dependency Injection)这个概念的兴起已经有很长时间了,把这个概念融入到框架中达到出神入化境地的,非Spring莫属.然而在前端领域,似乎很少会提到这个概念,难道 ...

  8. R语言爬虫:CSS方法与XPath方法对比(表格介绍)

    css 选择器与 xpath 用法对比 目标 匹配节点 CSS 3 XPath 所有节点 ~ * //* 查找一级.二级.三级标题节点 <h1>,<h2>,<h3> ...

  9. react前置路由守卫

    react中一切皆组件-- 目标:自定义user界面的前置路由守卫,当用户点击要进入user组件时,路由守卫发起判断,如果条件满足则进入,否则跳转至login组件. .入口文件index.js中代码如 ...

  10. SQL Server 中对 FOR XML和FROM的转换处理

    在SQL Server中对XML的再操作转换: 方法1: --生成XML SELECT * FROM [T_BAS_预算科目] FOR XML PATH --把XML转成SQL表 declare @X ...