」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 擴展 Node.js 應用程式:技術、工具和最佳實踐

擴展 Node.js 應用程式:技術、工具和最佳實踐

發佈於2024-11-07
瀏覽:625

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]刪除
最新教學 更多>
  • C++中如何將獨占指針作為函數或構造函數參數傳遞?
    C++中如何將獨占指針作為函數或構造函數參數傳遞?
    在構造函數和函數中將唯一的指數管理為參數 unique pointers( unique_ptr [2啟示。通過值: base(std :: simelor_ptr n) :next(std :: move(n)){} 此方法將唯一指針的所有權轉移到函數/對象。指針的內容被移至功能中,在操作...
    程式設計 發佈於2025-05-21
  • eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    稱量()和ast.literal_eval()中的Python Security 在使用用戶輸入時,必須優先確保安全性。強大的Python功能Eval()通常是作為潛在解決方案而出現的,但擔心其潛在風險。本文深入研究了eval()和ast.literal_eval()之間的差異,突出顯示其安全性含義...
    程式設計 發佈於2025-05-21
  • 如何有效地選擇熊貓數據框中的列?
    如何有效地選擇熊貓數據框中的列?
    在處理數據操作任務時,在Pandas DataFrames 中選擇列時,選擇特定列的必要條件是必要的。在Pandas中,選擇列的各種選項。 選項1:使用列名 如果已知列索引,請使用ILOC函數選擇它們。請注意,python索引基於零。 df1 = df.iloc [:,0:2]#使用索引0和1 ...
    程式設計 發佈於2025-05-21
  • 如何在鼠標單擊時編程選擇DIV中的所有文本?
    如何在鼠標單擊時編程選擇DIV中的所有文本?
    在鼠標上選擇div文本單擊帶有文本內容,用戶如何使用單個鼠標單擊單擊div中的整個文本?這允許用戶輕鬆拖放所選的文本或直接複製它。 在單個鼠標上單擊的div元素中選擇文本,您可以使用以下Javascript函數: function selecttext(canduterid){ if(d...
    程式設計 發佈於2025-05-21
  • PHP與C++函數重載處理的區別
    PHP與C++函數重載處理的區別
    作為經驗豐富的C開發人員脫離謎題,您可能會遇到功能超載的概念。這個概念雖然在C中普遍,但在PHP中構成了獨特的挑戰。讓我們深入研究PHP功能過載的複雜性,並探索其提供的可能性。 在PHP中理解php的方法在PHP中,函數超載的概念(如C等語言)不存在。函數簽名僅由其名稱定義,而與他們的參數列表無關...
    程式設計 發佈於2025-05-21
  • 同實例無需轉儲複製MySQL數據庫方法
    同實例無需轉儲複製MySQL數據庫方法
    在同一實例上複製一個MySQL數據庫而無需轉儲在同一mySQL實例上複製數據庫,而無需創建InterMediate sqql script。以下方法為傳統的轉儲和IMPORT過程提供了更簡單的替代方法。 直接管道數據 MySQL手動概述了一種允許將mysqldump直接輸出到MySQL cli...
    程式設計 發佈於2025-05-21
  • 如何修復\“常規錯誤:2006 MySQL Server在插入數據時已經消失\”?
    如何修復\“常規錯誤:2006 MySQL Server在插入數據時已經消失\”?
    How to Resolve "General error: 2006 MySQL server has gone away" While Inserting RecordsIntroduction:Inserting data into a MySQL database can...
    程式設計 發佈於2025-05-21
  • 為什麼我的CSS背景圖像出現?
    為什麼我的CSS背景圖像出現?
    故障排除:CSS背景圖像未出現 ,您的背景圖像儘管遵循教程說明,但您的背景圖像仍未加載。圖像和样式表位於相同的目錄中,但背景仍然是空白的白色帆布。 而不是不棄用的,您已經使用了CSS樣式: bockent {背景:封閉圖像文件名:背景圖:url(nickcage.jpg); 如果您的html,cs...
    程式設計 發佈於2025-05-21
  • 如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    postgresql:為每個唯一標識符在postgresql中提取最後一行,您可能需要遇到與數據集合中每個不同標識的信息相關的信息。考慮以下數據:[ 1 2014-02-01 kjkj 在數據集中的每個唯一ID中檢索最後一行的信息,您可以在操作員上使用Postgres的有效效率: id dat...
    程式設計 發佈於2025-05-21
  • 為什麼不使用CSS`content'屬性顯示圖像?
    為什麼不使用CSS`content'屬性顯示圖像?
    在Firefox extemers屬性為某些圖像很大,&& && && &&華倍華倍[華氏華倍華氏度]很少見,卻是某些瀏覽屬性很少,尤其是特定於Firefox的某些瀏覽器未能在使用內容屬性引用時未能顯示圖像的情況。這可以在提供的CSS類中看到:。 googlepic { 內容:url(&...
    程式設計 發佈於2025-05-21
  • Java是否允許多種返回類型:仔細研究通用方法?
    Java是否允許多種返回類型:仔細研究通用方法?
    在Java中的多個返回類型:一種誤解類型:在Java編程中揭示,在Java編程中,Peculiar方法簽名可能會出現,可能會出現,使開發人員陷入困境,使開發人員陷入困境。 getResult(string s); ,其中foo是自定義類。該方法聲明似乎擁有兩種返回類型:列表和E。但這確實是如此嗎...
    程式設計 發佈於2025-05-21
  • 使用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-21
  • 如何從PHP中的Unicode字符串中有效地產生對URL友好的sl。
    如何從PHP中的Unicode字符串中有效地產生對URL友好的sl。
    為有效的slug生成首先,該函數用指定的分隔符替換所有非字母或數字字符。此步驟可確保slug遵守URL慣例。隨後,它採用ICONV函數將文本簡化為us-ascii兼容格式,從而允許更廣泛的字符集合兼容性。 接下來,該函數使用正則表達式刪除了不需要的字符,例如特殊字符和空格。此步驟可確保slug僅包...
    程式設計 發佈於2025-05-21
  • 如何在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-21
  • 如何從PHP中的數組中提取隨機元素?
    如何從PHP中的數組中提取隨機元素?
    從陣列中的隨機選擇,可以輕鬆從數組中獲取隨機項目。考慮以下數組:; 從此數組中檢索一個隨機項目,利用array_rand( array_rand()函數從數組返回一個隨機鍵。通過將$項目數組索引使用此鍵,我們可以從數組中訪問一個隨機元素。這種方法為選擇隨機項目提供了一種直接且可靠的方法。
    程式設計 發佈於2025-05-21

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3