1. 引言
1.1 什么是Node.js
- Node.js简介:Node.js是一个基于Chrome V8引擎的JavaScript运行时,用于构建快速、可扩展的网络应用。
- Node.js的历史背景和发展:Node.js最初由Ryan Dahl在2009年发布,旨在解决I/O密集型应用的性能问题。随着时间的推移,Node.js社区不断壮大,提供了丰富的库和工具,使其成为构建现代Web应用的重要选择。
1.2 为什么选择Node.js
- 非阻塞I/O模型:Node.js使用事件驱动和非阻塞I/O模型,能够高效地处理大量并发连接。
- 使用JavaScript的优势:前后端使用相同的编程语言,简化开发流程,提高开发效率。
2. Node.js架构
2.1 单线程事件循环
- 事件循环的工作原理:Node.js使用单线程事件循环来处理异步操作。事件循环不断检查任务队列,执行回调函数。
- 非阻塞I/O模型:Node.js通过libuv库实现非阻塞I/O操作,避免了线程阻塞,提高了性能。
2.2 V8引擎
- V8引擎简介:V8是Google开发的高性能JavaScript引擎,广泛用于Chrome浏览器和Node.js。
- V8引擎的优化机制:V8通过JIT编译、隐藏类、内联缓存等技术优化JavaScript代码的执行性能。
2.3 libuv库
- libuv简介:libuv是一个跨平台的异步I/O库,负责处理文件系统、网络、子进程等操作。
- libuv的主要功能:提供事件循环、线程池、信号处理等功能。
2.4 Node.js模块系统
-
CommonJS模块:CommonJS是一种模块化规范,用于在Node.js中组织代码。
// 导出模块 module.exports = {add: function(a, b) {return a + b;} };// 导入模块 const math = require('./math'); console.log(math.add(2, 3)); // 输出: 5
-
ES6模块:ES6引入了模块化规范,使用
import
和export
关键字。// 导出模块 export function add(a, b) {return a + b; }// 导入模块 import { add } from './math.js'; console.log(add(2, 3)); // 输出: 5
2.5 Node.js核心模块
- 常用核心模块介绍:
-
fs
:文件系统操作const fs = require('fs'); fs.readFile('example.txt', 'utf8', (err, data) => {if (err) throw err;console.log(data); });
-
http
:创建HTTP服务器const http = require('http'); const server = http.createServer((req, res) => {res.statusCode = 200;res.setHeader('Content-Type', 'text/plain');res.end('Hello, World!\n'); }); server.listen(3000, '127.0.0.1', () => {console.log('Server running at http://127.0.0.1:3000/'); });
-
path
:处理和转换文件路径const path = require('path'); console.log(path.join(__dirname, 'file.txt')); // 输出: /path/to/current/directory/file.txt
-
3. 异步编程
3.1 回调函数
-
回调函数的基本用法:回调函数用于处理异步操作的结果。
function fetchData(callback) {setTimeout(() => {callback(null, 'Data fetched');}, 1000); }fetchData((err, data) => {if (err) throw err;console.log(data); // 输出: Data fetched });
-
回调地狱问题:多个嵌套的回调函数导致代码难以维护。
fetchData((err, data1) => {if (err) throw err;fetchData((err, data2) => {if (err) throw err;fetchData((err, data3) => {if (err) throw err;console.log(data1, data2, data3);});}); });
3.2 Promise
-
Promise的基本用法:Promise用于处理异步操作的结果,避免回调地狱。
function fetchData() {return new Promise((resolve, reject) => {setTimeout(() => {resolve('Data fetched');}, 1000);}); }fetchData().then(data => console.log(data)) // 输出: Data fetched.catch(err => console.error(err));
-
Promise链:多个Promise可以链式调用,处理多个异步操作。
fetchData().then(data1 => {console.log(data1); // 输出: Data fetchedreturn fetchData();}).then(data2 => {console.log(data2); // 输出: Data fetched}).catch(err => console.error(err));
3.3 async/await
-
async/await的基本用法:async/await是基于Promise的语法糖,使异步代码更易读。
async function fetchData() {return new Promise((resolve, reject) => {setTimeout(() => {resolve('Data fetched');}, 1000);}); }async function getData() {try {const data = await fetchData();console.log(data); // 输出: Data fetched} catch (err) {console.error(err);} }getData();
-
处理错误:使用try-catch块处理异步操作中的错误。
async function getData() {try {const data1 = await fetchData();console.log(data1); // 输出: Data fetchedconst data2 = await fetchData();console.log(data2); // 输出: Data fetched} catch (err)