- Javascript 基础教程
- Javascript - 首页
- JavaScript - 路线图
- JavaScript - 概述
- JavaScript - 特性
- JavaScript - 启用
- JavaScript - 位置
- JavaScript - 语法
- JavaScript - Hello World
- JavaScript - Console.log()
- JavaScript - 注释
- JavaScript - 变量
- JavaScript - let 语句
- JavaScript - 常量
- JavaScript - 数据类型
- JavaScript - 类型转换
- JavaScript - 严格模式
- JavaScript - 保留关键字
- JavaScript 运算符
- JavaScript - 运算符
- JavaScript - 算术运算符
- JavaScript - 比较运算符
- JavaScript - 逻辑运算符
- JavaScript - 位运算符
- JavaScript - 赋值运算符
- JavaScript - 条件运算符
- JavaScript - typeof 运算符
- JavaScript - 空值合并运算符
- JavaScript - delete 运算符
- JavaScript - 逗号运算符
- JavaScript - 分组运算符
- JavaScript - yield 运算符
- JavaScript - 展开运算符
- JavaScript - 指数运算符
- JavaScript - 运算符优先级
- JavaScript 控制流
- JavaScript - If...Else
- JavaScript - While 循环
- JavaScript - For 循环
- JavaScript - For...in
- Javascript - For...of
- JavaScript - 循环控制
- JavaScript - Break 语句
- JavaScript - Continue 语句
- JavaScript - Switch Case
- JavaScript - 用户定义迭代器
- JavaScript 函数
- JavaScript - 函数
- JavaScript - 函数表达式
- JavaScript - 函数参数
- JavaScript - 默认参数
- JavaScript - Function() 构造函数
- JavaScript - 函数提升
- JavaScript - 自执行函数
- JavaScript - 箭头函数
- JavaScript - 函数调用
- JavaScript - Function call()
- JavaScript - Function apply()
- JavaScript - Function bind()
- JavaScript - 闭包
- JavaScript - 变量作用域
- JavaScript - 全局变量
- JavaScript - 智能函数参数
- JavaScript 对象
- JavaScript - Number
- JavaScript - Boolean
- JavaScript - Strings
- JavaScript - Arrays
- JavaScript - Date
- JavaScript - DataView
- JavaScript - Handler
- JavaScript - Math
- JavaScript - RegExp
- JavaScript - Symbol
- JavaScript - Sets
- JavaScript - WeakSet
- JavaScript - Maps
- JavaScript - WeakMap
- JavaScript - 可迭代对象
- JavaScript - Reflect
- JavaScript - TypedArray
- JavaScript - 模板字面量
- JavaScript - 带标签的模板
- 面向对象 JavaScript
- JavaScript - 对象
- JavaScript - 类
- JavaScript - 对象属性
- JavaScript - 对象方法
- JavaScript - 静态方法
- JavaScript - 显示对象
- JavaScript - 对象访问器
- JavaScript - 对象构造函数
- JavaScript - 原生原型
- JavaScript - ES5 对象方法
- JavaScript - 封装
- JavaScript - 继承
- JavaScript - 抽象
- JavaScript - 多态
- JavaScript - 解构赋值
- JavaScript - 对象解构
- JavaScript - 数组解构
- JavaScript - 嵌套解构
- JavaScript - 可选链
- JavaScript - 全局对象
- JavaScript - Mixins
- JavaScript - 代理
- JavaScript 版本
- JavaScript - 历史
- JavaScript - 版本
- JavaScript - ES5
- JavaScript - ES6
- ECMAScript 2016
- ECMAScript 2017
- ECMAScript 2018
- ECMAScript 2019
- ECMAScript 2020
- ECMAScript 2021
- ECMAScript 2022
- JavaScript 异步
- JavaScript - 异步
- JavaScript - 回调函数
- JavaScript - Promises
- JavaScript - Async/Await
- JavaScript - 微任务
- JavaScript - Promise 化
- JavaScript - Promises 链式调用
- JavaScript - 定时事件
- JavaScript - setTimeout()
- JavaScript - setInterval()
- JavaScript Cookies
- JavaScript - Cookies
- JavaScript - Cookie 属性
- JavaScript - 删除 Cookies
- JavaScript 浏览器 BOM
- JavaScript - 浏览器对象模型
- JavaScript - Window 对象
- JavaScript - Document 对象
- JavaScript - Screen 对象
- JavaScript - History 对象
- JavaScript - Navigator 对象
- JavaScript - Location 对象
- JavaScript - Console 对象
- JavaScript Web APIs
- JavaScript - Web API
- JavaScript - History API
- JavaScript - Storage API
- JavaScript - Forms API
- JavaScript - Worker API
- JavaScript - Fetch API
- JavaScript - Geolocation API
- JavaScript 事件
- JavaScript - 事件
- JavaScript - DOM 事件
- JavaScript - addEventListener()
- JavaScript - 鼠标事件
- JavaScript - 键盘事件
- JavaScript - 表单事件
- JavaScript - Window/Document 事件
- JavaScript - 事件委托
- JavaScript - 事件冒泡
- JavaScript - 事件捕获
- JavaScript - 自定义事件
- JavaScript 错误处理
- JavaScript - 错误处理
- JavaScript - try...catch
- JavaScript - 调试
- JavaScript - 自定义错误
- JavaScript - 扩展错误
- JavaScript 重要关键字
- JavaScript - this 关键字
- JavaScript - void 关键字
- JavaScript - new 关键字
- JavaScript - var 关键字
- JavaScript HTML DOM
- JavaScript - HTML DOM
- JavaScript - DOM 方法和属性
- JavaScript - DOM Document
- JavaScript - DOM 元素
- JavaScript - DOM 属性 (Attr)
- JavaScript - DOM 表单
- JavaScript - 更改 HTML
- JavaScript - 更改 CSS
- JavaScript - DOM 动画
- JavaScript - DOM 导航
- JavaScript - DOM 集合
- JavaScript - DOM NodeList
- JavaScript - DOM DOMTokenList
- JavaScript 杂项
- JavaScript - Ajax
- JavaScript - 异步迭代
- JavaScript - Atomics 对象
- JavaScript - Rest 参数
- JavaScript - 页面重定向
- JavaScript - 对话框
- JavaScript - 页面打印
- JavaScript - 验证
- JavaScript - 动画
- JavaScript - 多媒体
- JavaScript - 图像地图
- JavaScript - 浏览器
- JavaScript - JSON
- JavaScript - 多行字符串
- JavaScript - 日期格式
- JavaScript - 获取日期方法
- JavaScript - 设置日期方法
- JavaScript - 模块
- JavaScript - 动态导入
- JavaScript - BigInt
- JavaScript - Blob
- JavaScript - Unicode
- JavaScript - 浅拷贝
- JavaScript - 调用栈
- JavaScript - 引用类型
- JavaScript - IndexedDB
- JavaScript - 点击劫持攻击
- JavaScript - 柯里化
- JavaScript - 图形
- JavaScript - Canvas
- JavaScript - 防抖
- JavaScript - 性能
- JavaScript - 样式指南
JavaScript - 回调函数
什么是回调函数?
JavaScript 中的回调函数通常作为另一个函数的参数传递。不要将“回调”视为此处名称或关键字。“回调”函数名称可以是任何有效的标识符。
回调函数可以在父函数完成父函数中的特定任务后在父函数内部被调用。它主要用于处理异步操作。
语法
您可以按照以下语法使用回调函数。
function func_name(callback) { // function body callback(); } func_name(callback); // Function invocation OR func_name(() => { // Callback function body })
在上述语法中,我们将“callback”作为 func_name() 函数的参数传递。
如上述语法所示,您还可以将箭头函数或匿名函数作为回调函数传递。
示例
在下面的代码中,我们将 multiply() 函数作为 sum() 函数的参数传递。
在 sum() 函数中,我们在最后调用回调函数。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); function multiply(a) { let m = a * 4; output.innerHTML = "The result is " + m + ".<br>"; } function sum(a, b, callback) { let c = a + b; callback(c); // Invoking the callback funciton } sum(4, 8, multiply); // Passing multiply function as a callback </script> </body> </html>
输出
The result is 48.
将匿名函数作为回调传递
示例
在下面的代码中,我们定义了 mathOperations() 函数,该函数将回调函数作为参数。
我们在 mathOperations() 函数内部调用回调函数并获取其返回值。
在调用 mathOperations() 函数时,我们传递了不同的匿名函数作为参数。这样,您可以使用回调函数控制要在特定函数内部执行的函数。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); function mathOperation(a, b, callback) { let result = callback(a, b); output.innerHTML += "The result is " + result + ".<br>"; } mathOperation(10, 20, function (a, b) { return a + b; // Callback function to add numbers }); mathOperation(20, 10, function (a, b) { return a - b; // Callback function to subtract numbers }); mathOperation(10, 20, function (a, b) { return a * b; // Callback function to multiply numbers }); </script> </body> </html>
输出
The result is 30. The result is 10. The result is 200.
回调函数的必要性
现在,让我们了解回调函数在实时开发中的必要性。
JavaScript 是一种单线程编程语言。因此,它逐行执行代码。当您需要从 API 获取数据、加载图像或执行任何异步操作时,它可能需要时间并阻塞其他代码的执行。
在这种情况下,您可以使用回调函数来执行必须在异步操作之后执行的代码,并且您可以执行其他代码而不会阻塞它。
例如,您正在发出 API 请求,并且需要 API 数据用于验证目的。因此,您可以在回调函数中执行数据验证并继续运行其他任务。
让我们使用 setTimeOut() 方法来理解它。
示例
在下面的代码中,我们使用 setTimeOut() 方法编写异步代码。
它在延迟 500 毫秒后执行 printMessage() 函数。我们将 printMessage() 函数作为 setTimeOut() 方法的回调传递。
在输出中,您可以观察到脚本在没有阻塞的情况下运行,并且它在 500 毫秒后执行 printMessage() 函数。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); output.innerHTML += "Start of the program. <br>"; setTimeout(printMessage, 500); // Asynchronous code function printMessage() { output.innerHTML += "In the printMessage() function. <br>"; } output.innerHTML += "End of the program. <br>"; </script> </body> </html>
输出
Start of the program. End of the program. In the printMessage() function.
带有内置方法的回调函数
许多内置 JavaScript 方法将回调函数作为参数,以在方法执行完成后执行自定义 JavaScript 代码。
在这里,我们将查看 2 到 3 个内置方法,这些方法将回调函数作为参数并附带示例。
带有回调函数的 JavaScript array.sort() 方法
array.sort() 方法用于对数组元素进行排序。默认情况下,它按升序对数组元素进行排序。如果要按降序或任何自定义顺序对数组元素进行排序,则可以将回调函数作为参数传递。
语法
请按照以下语法使用 array.sort() 方法
arr.sort(callback);
array.sort() 方法可以选择性地将回调函数作为参数。回调函数应返回 0、1 或 -1。
示例
在下面的代码中,我们定义了包含数字值的数组。首先,我们使用了没有回调函数的 sort() 方法。您可以看到它按升序对数组进行排序。
之后,我们将匿名函数作为 sort() 方法的回调函数传递。回调函数返回元素 b 和 a 之间的差值,以便按降序对数组进行排序。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); let arr = [23, 21, 56, 11, 10, 7, 8]; output.innerHTML += "The sorted array is - " + arr.sort(); // Sorting array in descending order let sorted = arr.sort(function (a, b) { return b - a; }); output.innerHTML += "<br>The sorted array in descending order is - " + sorted; </script> </body> </html>
输出
The sorted array is - 10,11,21,23,56,7,8 The sorted array in descending order is - 56,23,21,11,10,8,7
带有回调函数的 JavaScript array.filter() 方法
array.filter()
方法用于过滤数组元素。它接收一个回调函数作为参数。如果回调函数返回 true
,则过滤该元素。否则,跳过该数组元素。
语法
请按照以下语法使用 array.filter()
方法。
Array.filter(callback);
回调函数必须返回布尔值。
示例
在下面的代码中,我们将 filterCallback()
函数作为 filter()
方法的回调函数传递。如果数字为偶数,则 filterCallback()
函数返回布尔值。
最后,您可以在输出中看到过滤后的偶数。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); let arr = [23, 21, 56, 11, 10, 7, 8]; let eventNums = arr.filter(filtercallback); function filtercallback(element) { return element % 2 == 0; } output.innerHTML += "The original array is: " + arr + "<br>"; output.innerHTML += "The even numbers are: " + eventNums; </script> </body> </html>
输出
The original array is: 23,21,56,11,10,7,8 The even numbers are: 56,10,8
带有事件的回调函数
您可以在 JavaScript 中使用 addEventListner()
方法来监听事件。addEventListener()
方法将回调函数作为第二个参数。
每当网页上触发指定的事件时,它都会执行回调函数。
语法
请按照以下语法使用 addEventListener()
方法。
Element.addEventListener(event, callback);
在上面的语法中,event
是一个表示事件名称的字符串,callback
是一个在事件触发时应执行的函数。
示例
在下面的代码中,我们创建了一个按钮。
在 JavaScript 中,我们使用其 ID 访问了按钮并添加了点击事件。
每当用户点击按钮时,它都会打印消息。
<html> <body> <button id = "btn"> Click Me </button> <p id = "output"> </p> <script> let output = document.getElementById('output'); let button = document.getElementById('btn'); button.addEventListener('click', function () { output.innerHTML = 'You have clicked the button. <br>'; }); </script> </body> </html>
输出
嵌套回调和回调地狱
您可以像 JavaScript 中的嵌套循环或 if-else 语句一样拥有嵌套回调函数。如果第一个函数依赖于第二个函数的数据,而第二个函数依赖于第三个函数的数据,则您可能需要嵌套回调函数。
让我们通过下面的示例来理解它。
示例
asyncTask()
函数在下面的代码中完成任务并调用作为参数传递的回调函数。
之后,我们调用了 asyncTask()
函数并将回调函数作为第三个参数传递。在回调函数中,我们再次调用了 asyncTask()
函数并将回调函数作为第三个参数传递。
这样,我们在 3 个嵌套级别使用了回调函数。
<html> <body> <div id = "output"> </div> <script> let output = document.getElementById('output'); function asyncTask(taskName, duration, callback) { output.innerHTML += "Task started " + taskName + "<br/>" setTimeout(() => { output.innerHTML += 'Completed ' + taskName + '<br/>'; callback(); }, duration); } // Task 1 asyncTask('Task 1', 1000, () => { // Task 2 asyncTask('Task 2', 1500, () => { // Task 3 asyncTask('Task 3', 1000, () => { output.innerHTML += "All tasks completed"; }); }); }); </script> </body> </html>
输出
Task started Task 1 Completed Task 1 Task started Task 2 Completed Task 2 Task started Task 3 Completed Task 3 All tasks completed
由于其嵌套回调的复杂语法,它也被称为回调地狱。
每当您需要使用嵌套回调函数时,可以使用 Promise 或 async/await 来编写更简单的代码。
在接下来的章节中,您将学习 Promise 和 async/await 来处理异步操作。