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

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

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

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]删除
最新教程 更多>
  • 如何使用不同数量列的联合数据库表?
    如何使用不同数量列的联合数据库表?
    合并列数不同的表 当尝试合并列数不同的数据库表时,可能会遇到挑战。一种直接的方法是在列数较少的表中,为缺失的列追加空值。 例如,考虑两个表,表 A 和表 B,其中表 A 的列数多于表 B。为了合并这些表,同时处理表 B 中缺失的列,请按照以下步骤操作: 确定表 B 中缺失的列,并将它们添加到表的末...
    编程 发布于2025-05-20
  • 为什么使用Firefox后退按钮时JavaScript执行停止?
    为什么使用Firefox后退按钮时JavaScript执行停止?
    导航历史记录问题:JavaScript使用Firefox Back Back 此行为是由浏览器缓存JavaScript资源引起的。要解决此问题并确保在后续页面访问中执行脚本,Firefox用户应设置一个空功能。 警报'); }; alert('inline Alert')...
    编程 发布于2025-05-20
  • 如何在php中使用卷发发送原始帖子请求?
    如何在php中使用卷发发送原始帖子请求?
    如何使用php 创建请求来发送原始帖子请求,开始使用curl_init()开始初始化curl session。然后,配置以下选项: curlopt_url:请求 [要发送的原始数据指定内容类型,为原始的帖子请求指定身体的内容类型很重要。在这种情况下,它是文本/平原。要执行此操作,请使用包含以下标头...
    编程 发布于2025-05-20
  • Android如何向PHP服务器发送POST数据?
    Android如何向PHP服务器发送POST数据?
    在android apache httpclient(已弃用) httpclient httpclient = new defaulthttpclient(); httppost httppost = new httppost(“ http://www.yoursite.com/script.p...
    编程 发布于2025-05-20
  • Python中何时用"try"而非"if"检测变量值?
    Python中何时用"try"而非"if"检测变量值?
    使用“ try“ vs.” if”来测试python 在python中的变量值,在某些情况下,您可能需要在处理之前检查变量是否具有值。在使用“如果”或“ try”构建体之间决定。“ if” constructs result = function() 如果结果: 对于结果: ...
    编程 发布于2025-05-20
  • Python环境变量的访问与管理方法
    Python环境变量的访问与管理方法
    Accessing Environment Variables in PythonTo access environment variables in Python, utilize the os.environ object, which represents a mapping of envir...
    编程 发布于2025-05-20
  • 如何在鼠标单击时编程选择DIV中的所有文本?
    如何在鼠标单击时编程选择DIV中的所有文本?
    在鼠标上选择div文本单击带有文本内容,用户如何使用单个鼠标单击单击div中的整个文本?这允许用户轻松拖放所选的文本或直接复制它。 在单个鼠标上单击的div元素中选择文本,您可以使用以下Javascript函数: function selecttext(canduterid){ if(do...
    编程 发布于2025-05-20
  • 如何简化PHP中的JSON解析以获取多维阵列?
    如何简化PHP中的JSON解析以获取多维阵列?
    php 试图在PHP中解析JSON数据的JSON可能具有挑战性,尤其是在处理多维数组时。 To simplify the process, it's recommended to parse the JSON as an array rather than an object.To do...
    编程 发布于2025-05-20
  • 为什么Microsoft Visual C ++无法正确实现两台模板的实例?
    为什么Microsoft Visual C ++无法正确实现两台模板的实例?
    The Mystery of "Broken" Two-Phase Template Instantiation in Microsoft Visual C Problem Statement:Users commonly express concerns that Micro...
    编程 发布于2025-05-20
  • 如何在Java中正确显示“ DD/MM/YYYY HH:MM:SS.SS”格式的当前日期和时间?
    如何在Java中正确显示“ DD/MM/YYYY HH:MM:SS.SS”格式的当前日期和时间?
    如何在“ dd/mm/yyyy hh:mm:mm:ss.ss”格式“ gormat 解决方案:的,请访问量很大,并应为procectiquiestate的,并在整个代码上正确格式不多: java.text.simpledateformat; 导入java.util.calendar; 导入java...
    编程 发布于2025-05-20
  • 在UTF8 MySQL表中正确将Latin1字符转换为UTF8的方法
    在UTF8 MySQL表中正确将Latin1字符转换为UTF8的方法
    在UTF8表中将latin1字符转换为utf8 ,您遇到了一个问题,其中含义的字符(例如,“jáuòiñe”)在utf8 table tabled tablesset中被extect(例如,“致电。为了解决此问题,您正在尝试使用“ mb_convert_encoding”和“ iconv”转换受...
    编程 发布于2025-05-20
  • 如何使用node-mysql在单个查询中执行多个SQL语句?
    如何使用node-mysql在单个查询中执行多个SQL语句?
    在node-mysql node-mysql文档最初出于安全原因最初禁用多个语句支持,因为它可能导致SQL注入攻击。要启用此功能,您需要在创建连接时将倍增设置设置为true: var connection = mysql.createconnection({{multipleStatement:...
    编程 发布于2025-05-20
  • 使用jQuery如何有效修改":after"伪元素的CSS属性?
    使用jQuery如何有效修改":after"伪元素的CSS属性?
    在jquery中了解伪元素的限制:访问“ selector 尝试修改“:”选择器的CSS属性时,您可能会遇到困难。 This is because pseudo-elements are not part of the DOM (Document Object Model) and are th...
    编程 发布于2025-05-20
  • 如何在Java字符串中有效替换多个子字符串?
    如何在Java字符串中有效替换多个子字符串?
    在java 中有效地替换多个substring,需要在需要替换一个字符串中的多个substring的情况下,很容易求助于重复应用字符串的刺激力量。 However, this can be inefficient for large strings or when working with nu...
    编程 发布于2025-05-20
  • Java为何无法创建泛型数组?
    Java为何无法创建泛型数组?
    通用阵列创建错误 arrayList [2]; JAVA报告了“通用数组创建”错误。为什么不允许这样做?答案:Create an Auxiliary Class:public static ArrayList<myObject>[] a = new ArrayList<myO...
    编程 发布于2025-05-20

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

Copyright© 2022 湘ICP备2022001581号-3