网站首页 > 技术文章 正文
在这篇文章中,作者总结了 3 个在 JavaScript 面试问题中问得最多的问题(不清楚国内是不是)。这三个问题不是关于任何库的用法或 ES6 的新功能,而主要是对原生 JavaScript 的理解。
Question #1: 事件代理
当创建一个应用时,不可避免的会遇到监听事件触发的需求。这里有一个小的简单的待办列表要完成,想要在用户点击其中一个列表项时触发一个动作。下面是一段 HTML 代码:
<ul id="todo-app">
你可能会想像下面这样来写:
document.addEventListener('DOMContentLoaded', function() {
当然这样写在技术上是完全可以的,唯一的问题就是当列表项过多的时候(比如 10,000 个),你的这段函数就将会同时创建 10,000 个监听函数。
下面是一种更有效率的写法:
document.addEventListener('DOMContentLoaded', function() {
Question #2: 在循环中使用闭包
闭包是 JavaScript 的一个重要特性,开发者可以用来模拟私有方法。在这里有个简单的问题:
实现一个函数,循环遍历整数列表,并在 3 秒后打印每个元素的索引。
一个常见的错误实现:
const arr = [10, 12, 15, 21];for (var i = 0; i < arr.length; i++) {
如果你执行这段代码,会发现每次输出的是 4 而不是按顺序的 0,1,2,3。
原因在于 setTimeout 创建了一个匿名函数并访问处于外部的变量 i,都处于同一环境中。当 console.log 被调用的时候,匿名函数保持对外部变量 i 的引用,此时 for 循环已经结束, i 的值被修改成了 4。为了得到想要的结果,需要在每次循环中创建变量 i 的拷贝。
事实上正确的写法有好几种,这里列举用得最多的两种:
const arr = [10, 12, 15, 21];for (var i = 0; i < arr.length; i++) {
const arr = [10, 12, 15, 21];for (let i = 0; i < arr.length; i++) {
Question #3: Debouncing
有一些浏览器事件可以在很短的时间内快速启动多次,例如调整窗口大小或滚动页面。如果你在窗口滚动上绑定了事件,那么可能在用户滚动页面的几秒钟里,你的事件方法就执行了数千次,这就会导致很严重的性能问题。
一个真实的案例就是 2011 年 Twitter,在你滚动 Twitter feed 时,其会变得非常慢甚至未响应。这里有一篇 blog 就详细讲了当时的这个 bug,也就是在 scroll 事件上绑定一个复杂函数是多糟的主意。
Debouncing 就是解决这个问题的一种方法,简单来说就是限制函数调用的间隔时间。如果在时间间隔内再次触发事件,就重启定时器并忽略掉这次事件。
// debounce function that will wrap our eventfunction debounce(fn, delay) {
使用:
// function to be called when user scrollsfunction foo() {
与 Debouncing 类似的技术是 Throttling,同样也是使用计时器来控制事件的触发,不同之处在于 Throttling 没有忽略掉事件,而是延迟触发。如果想了解更多,可以进一步阅读下面的这几篇文章。
猜你喜欢
- 2024-10-18 前端反向代理(前端反向代理怎么配置)
- 2024-10-18 JavaScript 九种跨域方式实现原理
- 2024-10-18 Proxy 来代理 JavaScript 里的类(js中proxy)
- 2024-10-18 octokit.js:2023每个前端都值得学习的通用SDK!
- 2024-10-18 一面 2:JS-Web-API 知识点与高频考题解析
- 2024-10-18 使用reveal.js制作精美的网页版PPT
- 2024-10-18 es6中的代理-Proxy(es proxy)
- 2024-10-18 事件——《JS高级程序设计》(javascript高级程序设计 javascript权威指南)
- 2024-10-18 「JavaScript 从入门到精通」18.WebApi介绍
- 2024-10-18 day7:前端面试题(js)(前端面试题2021及答案js)
- 最近发表
- 标签列表
-
- cmd/c (64)
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- sqlset (64)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- chromepost (65)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- linux删除一个文件夹 (65)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)