”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 扩展 Node.js 应用程序:技术、工具和最佳实践

扩展 Node.js 应用程序:技术、工具和最佳实践

发布于2024-11-07
浏览:909

Scaling Node.js Applications: Techniques, Tools, and Best Practices

随着 Node.js 应用程序的增长,对更好的性能和可扩展性的需求也在增加。 Node.js 旨在处理大规模数据密集型应用程序,但了解如何正确扩展它对于在负载下保持性能和可用性至关重要。在本文中,我们将介绍有效扩展 Node.js 应用程序的关键技术和工具。

为什么要扩展 Node.js 应用程序?

扩展是指应用程序处理不断增加的负载的能力——无论是由于用户群的增长、更多的数据还是更高的流量。如果不进行扩展,应用程序可能会面临性能低下、停机和资源效率低下的问题。

两种缩放类型

  1. 垂直扩展:为单个服务器添加更多功能(CPU、RAM)。虽然这增加了服务器容量,但它有物理限制。

  2. 水平扩展:添加更多服务器来分配负载,通常称为“横向扩展”。这种方法比较灵活,常用于大型系统。

扩展 Node.js 应用程序的关键技术

1.负载均衡

负载平衡是在多个服务器之间分配传入流量的做法,确保没有任何一台服务器被压垮。这在水平扩展中尤其重要,其中 Node.js 应用程序的多个实例正在运行。

示例:使用 NGINX 进行负载平衡

http {
    upstream node_servers {
        server 127.0.0.1:3000;
        server 127.0.0.1:3001;
        server 127.0.0.1:3002;
    }

    server {
        listen 80;
        location / {
            proxy_pass http://node_servers;
        }
    }
}

解释

  • upstream 块定义了多个 Node.js 实例。
  • 传入请求在实例之间分配,从而提高性能。

2. 聚类

Node.js 是单线程的,但 Cluster 模块允许您通过创建共享同一服务器端口的子进程来利用多个 CPU 核心。

示例:使用集群模块

const cluster = require('cluster');
const http = require('http');
const numCPUs = require('os').cpus().length;

if (cluster.isMaster) {
    // Fork workers.
    for (let i = 0; i  {
        console.log(`Worker ${worker.process.pid} died`);
    });
} else {
    // Workers can share the same port
    http.createServer((req, res) => {
        res.writeHead(200);
        res.end('Hello World');
    }).listen(8000);
}

解释

  • master进程创建与CPU核数相等的worker进程。
  • 每个工作人员处理传入的请求,有效地分配负载。

3. 缓存

缓存通过将频繁请求的数据存储在内存中,而不是从数据库中重新获取数据或重新计算结果,有助于缩短响应时间并减少负载。

示例:使用 Redis 进行缓存

const redis = require('redis');
const client = redis.createClient();

function cacheMiddleware(req, res, next) {
    const key = req.url;
    client.get(key, (err, data) => {
        if (err) throw err;
        if (data !== null) {
            res.send(data);
        } else {
            next();
        }
    });
}

app.get('/data', cacheMiddleware, (req, res) => {
    const data = getDataFromDatabase();
    client.setex(req.url, 3600, JSON.stringify(data));
    res.json(data);
});

解释

  • 中间件在进行数据库调用之前检查 Redis 的缓存数据。
  • 如果数据未缓存,则继续获取数据、缓存数据并发送响应。

4.无状态微服务

通过将单体 Node.js 应用程序分解为无状态微服务,您可以独立扩展每个服务。这可以确保扩展应用程序的一部分(例如用户身份验证)不会影响其他部分(例如支付处理)。

示例:微服务架构

  • 每个微服务(身份验证、产品目录、订单管理)都是独立部署的。
  • API 网关或服务网格处理将请求路由到正确的微服务。

5. 使用反向代理

反向代理服务器可以处理各种任务,例如负载平衡、SSL 终止和提供静态内容,从而减少 Node.js 服务器上的负载。

示例:使用 NGINX 提供静态内容

server {
    listen 80;

    location / {
        proxy_pass http://localhost:3000;
    }

    location /static/ {
        root /var/www/html;
    }
}

解释

  • NGINX 用于代理 Node.js 的动态请求并直接从服务器提供静态文件(CSS、JS、图像)。

用于扩展 Node.js 应用程序的工具

1.PM2.5

PM2 是 Node.js 应用程序的生产就绪流程管理器,支持集群、自动重启、负载平衡和流程监控。

示例:使用 PM2 扩展应用程序

# Start the application with cluster mode and 4 instances
pm2 start app.js -i 4

解释

  • PM2 管理应用程序的多个实例,提供自动负载平衡和进程监控。

2.Docker 和 Kubernetes

使用 Docker 容器化您的应用程序并将其部署在 Kubernetes 上,您可以轻松地跨多个服务器扩展 Node.js 应用程序。 Kubernetes 自动处理编排、负载平衡和扩展。

示例:Docker化 Node.js 应用程序

# Dockerfile
FROM node:14
WORKDIR /app
COPY package*.json ./
RUN npm install
COPY . .
EXPOSE 3000
CMD ["node", "app.js"]

扩展 Node.js 应用程序的最佳实践

  1. 监控性能:使用 New RelicDatadogPrometheus 等工具来跟踪性能指标并识别瓶颈。
  2. 使用异步编程:当异步处理 I/O 操作等任务时,Node.js 表现最佳。避免阻塞事件循环。
  3. 优化数据库查询:使用连接池、索引和缓存来减少数据库负载。
  4. 水平扩展优于垂直扩展:水平扩展(添加更多服务器)比垂直扩展(增加服务器资源)提供更多的灵活性和容错能力。
  5. 保持服务无状态:无状态服务更容易扩展,因为它们不依赖于请求之间的内存状态。使用 Redis 或数据库等外部系统进行会话管理。

结论

随着应用程序的增长,扩展 Node.js 应用程序对于保持性能至关重要。通过利用负载平衡、集群、缓存和无状态微服务等技术以及 PM2、Docker 和 Kubernetes 等工具,您可以确保 Node.js 应用程序高效扩展。实施这些策略将使您的应用程序能够处理增加的流量和更大的数据集,而不会影响速度或可靠性。

版本声明 本文转载于:https://dev.to/imsushant12/scaling-nodejs-applications-techniques-tools-and-best-practices-3344?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 如何高效更新 Firestore 中的对象数组?
    如何高效更新 Firestore 中的对象数组?
    更新 Firestore 中的对象数组更新 Firestore 中的对象数组可能是一个简单的过程,但它需要使用Firestore SDK 中的特定方法。要将新记录追加到对象数组中,可以使用 arrayUnion() 方法。此方法采用一个元素数组作为其参数,并将这些元素添加到现有数组中,但前提是它们尚...
    编程 发布于2024-11-07
  • 简单的命令行龙与地下城
    简单的命令行龙与地下城
    作为 Coding Nomads 训练营的参与者,Python 课程的早期项目之一是一个简单的《龙与地下城》命令行游戏。目标:找到一把剑并杀死一条龙。命令行游戏由用户输入驱动(响应所呈现的二元选择;例如,是/否、战斗/安全)。除了用户输入之外,标记变量在代码中也很重要,可以跟踪玩家在游戏中的位置(即...
    编程 发布于2024-11-07
  • 如何获取已安装的 Go 软件包的完整列表?
    如何获取已安装的 Go 软件包的完整列表?
    检索 Go 中已安装软件包的综合列表在多台计算机上传输 Go 软件包安装时,有必要获取详细的清单所有已安装的软件包。本文概述了此任务的简单且最新的解决方案。解决方案:利用“go list”与过时的答案相反,当前的建议列出 Go 中已安装的软件包是使用“go list”命令。通过指定三个文字句点 (&...
    编程 发布于2024-11-07
  • Offload - 一个统一的 javascript SDK,支持浏览器内 AI
    Offload - 一个统一的 javascript SDK,支持浏览器内 AI
    今天我想分享Offload,一个直接在用户浏览器上运行AI的javascript SDK。 什么是卸载? 这是一个 SDK,您可以使用它向您的网站添加 AI,但有一个特点:它允许您的用户在本地运行 AI 任务,将数据保存在设备上,从而无需发送数据到第三方推理 API。 此外,它降低了...
    编程 发布于2024-11-07
  • 扩展语法与其余参数:有什么区别?
    扩展语法与其余参数:有什么区别?
    扩展语法与剩余参数:了解差异在 ES2015 中,扩展语法和剩余参数这两个新功能提供了强大的方法操作数组和对象。虽然两者看起来相似,但它们具有不同的用途,并且各自具有自己独特的功能。扩展语法扩展语法(由三个点“...”表示) ) 允许您将一个可迭代对象(例如数组或对象)展开或扩展为另一个可迭代对象中...
    编程 发布于2024-11-07
  • 如何使用 CSS 让图像随浏览器大小自动调整大小?
    如何使用 CSS 让图像随浏览器大小自动调整大小?
    使用 CSS 根据浏览器大小自动调整图像大小您希望在调整浏览器窗口大小时自动调整图像大小,但提供的代码不工作。让我们研究一个解决方案。为了使图像灵活,您需要向它们添加 max-width: 100% 和 height: auto 。然而,IE8 有一个错误,这不起作用。要修复此问题,请为 IE8 添...
    编程 发布于2024-11-07
  • Next.js:Web 开发的 React 框架
    Next.js:Web 开发的 React 框架
    I'm thrilled to introduce Next.js, a game-changing React framework for web development. It makes it easy to build fast, server-rendered, and staticall...
    编程 发布于2024-11-07
  • 了解网络存储
    了解网络存储
    目录 曲奇饼 本地存储 会话存储 索引数据库 对比分析 安全考虑 结论 介绍 数据存储是现代 Web 应用程序的一个重要方面。无论是保存用户首选项、缓存数据以供离线使用,还是跟踪会话,在浏览器中管理数据的方式都会显着影响用户体验。我们有多种在浏览器中存储数据的选项,...
    编程 发布于2024-11-07
  • 指针如何影响 Go 函数中的值修改?
    指针如何影响 Go 函数中的值修改?
    理解 Go 中指针的值修改在 Go 中,指针允许间接访问和修改值。然而,在将指针传递给函数时,了解指针的工作原理至关重要。将指针传递给函数时,会出现两种情况:值修改与指针重新分配。场景 1 : 值修改考虑这段代码:type Test struct { Value int } func main() ...
    编程 发布于2024-11-07
  • 将 django 部署到生产环境
    将 django 部署到生产环境
    我最近将我自己的 django 应用程序部署到生产环境中。该网站名为 videoeiro.com,是用 django HTML/CSS/JS Tailwind 开发的。 设置 我正在使用 debian 12 服务器,它将通过 cloudflare 隧道公开我的应用程序。所有静态文件都...
    编程 发布于2024-11-07
  • 实施雪花 ID 生成器
    实施雪花 ID 生成器
    什么是雪花 ID? 雪花 ID 在分布式环境中用于生成无冲突、简短、唯一的 ID。与依赖数据库生成 ID 或使用长 128 位 UUID 等传统方法不同,Snowflake ID 使用时间和简单的按位运算。这种巧妙的技术允许每个微服务独立生成唯一的 ID,而不需要中央系统来避免冲突...
    编程 发布于2024-11-07
  • 如何在没有 JS 框架的情况下使用 CSS 设计 SVG 图像样式?
    如何在没有 JS 框架的情况下使用 CSS 设计 SVG 图像样式?
    使用 CSS 设置 SVG 图像样式:一种新颖的方法在本文中,我们将探索一种使用 CSS 嵌入 SVG 图像并操纵其外观的新颖方法,而无需使用 CSS使用 JS-SVG 框架。问题陈述以前,集成 SVG 图像同时通过 CSS 保持对其元素的访问一直是一个挑战。虽然 JS-SVG 框架提供了解决方案,...
    编程 发布于2024-11-07
  • ## 你能确定用户是否可以在 JavaScript 中单击后退按钮吗?
    ## 你能确定用户是否可以在 JavaScript 中单击后退按钮吗?
    浏览器历史记录导航:确定后退按钮可用性开发 Web 应用程序时,确定用户是否可以在他们的浏览器历史记录。然而,出于安全考虑,JavaScript 缺乏直接手段来确定浏览器历史记录是否存在。尝试的解决方案及其局限性:1。 History.previous:虽然该属性理论上提供了有关历史记录中上一页的信...
    编程 发布于2024-11-07
  • 如何在保持模糊背景的同时去除子元素的背景模糊?
    如何在保持模糊背景的同时去除子元素的背景模糊?
    从子元素中删除背景模糊您有一个 ,其背景图像应用了模糊效果。但是,所有子元素也会受到这种模糊的影响,这是不希望的。本文提供了解决此问题的解决方案,允许您在保持背景图像的模糊效果的同时保留子元素的清晰度。解决方案:创建叠加元素要实现此目的,您可以在父元素中创建一个单独的 并将背景图像和模糊效果应用到...
    编程 发布于2024-11-07
  • Leetcode:字符串的最大公约数
    Leetcode:字符串的最大公约数
    问题陈述 1071. 字符串的最大公约数 对于两个字符串 s 和 t,当且仅当 s = t t t ... t t (即 t 与自身连接一次或多次)时,我们才说“t 除 s”。 给定两个字符串 str1 和 str2,返回使 x 整除 str1 和 str2 的最大字符串 x。 ...
    编程 发布于2024-11-07

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3