什么是Node.js_如何使用它构建服务器端应用程序【教程】

Node.js是基于V8的JavaScript运行时,适合I/O密集型服务因事件驱动非阻塞I/O,但不适用于CPU密集型任务;单线程+事件循环避免上下文切换开销;内置模块多为异步,require同步加载;默认不支持ESM需配置;原生http模块可快速启服,须监听'0.0.0.0'并调用res.end();Express简化路由与中间件管理,但需注意中间件注册顺序。

Node.js 不是编程语言,而是一个基于 Chrome V8 引擎的 JavaScript 运行时环境,它让你能在服务器端直接运行 JavaScript 代码。

为什么 Node.js 适合写服务器?

它的核心优势在于事件驱动、非阻塞 I/O 模型——这意味着一个进程能同时处理成百上千个连接,特别适合 I/O 密集型服务(比如 API 接口、实时聊天、文件上传代理)。

但注意:Node.js 不适合 CPU 密集型任务(如视频转码、复杂计算),这类操作会阻塞整个事件循环。

  • 单线程 + 事件循环机制,避免了多线程上下文切换开销
  • require() 加载模块是同步的,但绝大多数内置模块(如 fshttp)提供的是异步接口
  • 默认不支持 ES modulesimport/export),除非加 "type": "module"package.json 或用 .mjs 后缀

如何快速启动一个 HTTP 服

务器?

不用任何框架,只用 Node.js 内置的 http 模块就能跑起来。这是最轻量、最贴近底层的理解方式。

常见错误:监听 localhost 却从外部网络访问失败;或没处理请求体导致 req 流未消费而卡住。

const http = require('http');
const url = require('url');

const server = http.createServer((req, res) => {
  const parsedUrl = url.parse(req.url, true);
  
  if (req.method === 'GET' && parsedUrl.pathname === '/') {
    res.writeHead(200, { 'Content-Type': 'text/plain' });
    res.end('Hello from Node.js!\n');
  } else {
    res.writeHead(404);
    res.end('Not found\n');
  }
});

server.listen(3000, '0.0.0.0', () => {
  console.log('Server running on http://localhost:3000');
});
  • 监听 '0.0.0.0' 而不是 'localhost',才能从本机以外访问(如 Docker 容器、局域网设备)
  • 必须调用 res.end(),否则客户端会一直等待响应结束
  • 如果要读取 POST 数据,需监听 req.on('data', ...)req.on('end', ...),不能跳过

用 Express 管理路由和中间件是否必要?

对简单脚本或原型来说,原生 http 足够;但只要涉及路径匹配、JSON 解析、静态文件、错误统一处理,Express 就几乎是标配。

容易踩的坑:app.use(express.json()) 必须在路由前注册;否则 req.body 始终为 undefined

const express = require('express');
const app = express();

// 这两行顺序不能错
app.use(express.json());
app.use(express.urlencoded({ extended: true }));

app.get('/', (req, res) => {
  res.json({ message: 'OK' });
});

app.post('/echo', (req, res) => {
  res.json(req.body); // 能拿到 JSON body 的前提是上面中间件已启用
});

app.listen(3000);
  • express.json() 只解析 Content-Type: application/json
  • express.urlencoded({ extended: true }) 支持嵌套对象(如 user[address][city]),设为 false 则只支持扁平键值对
  • 静态资源建议用 express.static('public'),别自己用 fs.readFile 发送文件——它不处理缓存头、范围请求、404 等细节

真正难的从来不是“怎么写个服务器”,而是怎么让这个服务器在真实场景中可靠运行:进程崩溃后自动重启、日志分级输出、环境变量隔离、请求超时控制、健康检查端点、与数据库连接池协同……这些都不是 http.createServer 一行代码能覆盖的。