”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Node.js 应用程序中的错误处理和日志记录

Node.js 应用程序中的错误处理和日志记录

发布于2024-11-09
浏览:822

Error Handling and Logging in Node.js Applications

介绍

在后端开发领域,可靠的错误处理和结构化日志记录对于构建有弹性和可维护的应用程序至关重要。 Node.js 中有效的错误处理不仅可以改善用户体验,还可以简化调试并增强应用程序监控。结合适当的日志记录,开发人员可以更快地跟踪问题并实时监控系统运行状况。在本文中,我们将深入探讨 Node.js 中的错误处理和日志记录策略,这些策略可以使应用程序更加健壮并可用于生产。

1.了解 Node.js 中的错误处理

Node.js 中的错误处理有其独特的挑战,主要是由于其异步、事件驱动的架构。让我们探讨 Node.js 中有效错误处理的一些关键区别和原则:

  • 同步与异步错误:同步错误立即发生,可以使用传统的 try-catch 块捕获,而异步错误稍后发生,需要不同的处理,例如回调、承诺或异步/等待模式。
  • 统一错误处理策略:通过大量的异步操作,维护统一的方法有助于更有效地管理整个应用程序中的错误。一致的错误处理简化了调试并确保统一传播和记录错误。

2.基本错误处理技术

Node.js 提供了多种管理错误的机制,特别是在异步工作流程中:

  • Try-Catch 块:这些对于处理同步代码中的错误很有用,但不会捕获异步代码中的错误。例如:
  try {
    const data = JSON.parse(jsonString);
  } catch (error) {
    console.error("JSON parsing error:", error);
  }
  • 错误对象:Node.js 中的错误对象携带堆栈跟踪等重要信息,可以帮助开发人员调试问题。自定义错误消息应该清晰且可操作。

  • Promises 和 Async/Await 中的错误处理:

    • Promises:使用 .catch() 处理 Promise 拒绝。
    fetchData()
      .then(data => console.log(data))
      .catch(error => console.error("Error fetching data:", error));
    
    • Async/Await:将await 调用包装在try-catch 块内以进行异步错误处理。
    async function fetchData() {
      try {
        const data = await someAsyncFunction();
        console.log(data);
      } catch (error) {
        console.error("Error:", error);
      }
    }
    
  • 全局错误处理

    • 未捕获的异常:使用process.on('uncaughtException',callback)捕获未捕获的异常。
    • 未处理的拒绝:使用 process.on('unhandledRejection',callback) 处理未处理的 Promise 拒绝。这提供了生产中的最后一道防线,尽管通常建议在错误发生的地方进行处理。

3.高级错误处理模式

为了实现更具可扩展性和可维护性的错误处理,这些先进技术至关重要:

  • Express 中的中间件错误处理: Express 提供了内置的错误处理机制,可以捕获错误并将其传递给自定义中间件。
  app.use((err, req, res, next) => {
    console.error("Error:", err);
    res.status(500).send("Something went wrong!");
  });
  • 集中错误管理: 可以创建集中式错误处理模块来定义自定义错误响应和消息,从而提供处理错误的标准化方法。
  class AppError extends Error {
    constructor(message, statusCode) {
      super(message);
      this.statusCode = statusCode;
      Error.captureStackTrace(this, this.constructor);
    }
  }
  • 错误代码及分类: 使用 HTTP 状态代码对错误进行分类(4xx 表示客户端错误,5xx 表示服务器错误),并添加自定义代码以进行更详细的日志记录和错误跟踪。

4.在 Node.js 中实现日志记录

日志记录可以深入了解应用程序中发生的情况,对于跟踪错误非常有价值。以下是如何在 Node.js 中实现有效的日志记录:

  • 基本控制台日志记录:
    控制台对象(console.log、console.error 等)很方便,但在生产中受到限制。对于结构化、基于级别的日志记录,最好使用专用的日志记录库。

  • 使用 Winston 进行日志记录:
    Winston 是一个功能强大的日志记录库,提供结构化日志记录,包括信息、警告、错误和调试等级别。

  const winston = require("winston");

  const logger = winston.createLogger({
    level: "info",
    format: winston.format.json(),
    transports: [
      new winston.transports.File({ filename: "error.log", level: "error" }),
      new winston.transports.Console({ format: winston.format.simple() })
    ]
  });
  • 日志级别:

    • info:有关应用程序行为的一般信息。
    • warn:警告,通常是非破坏性但值得注意的事件。
    • 错误:需要立即关注的关键问题。
    • debug:对调试有用的详细信息。
  • 旋转日志:

    日志轮换限制日志文件大小并确保高效存储。 Winston 的 winston-daily-rotate-file 传输可以帮助管理每天的日志。

5.生产中的错误和日志管理

管理生产中的错误和日志需要额外考虑以维护性能和数据安全。

  • 使用日志服务:
    集成 LogglyPapertrailELK Stack(Elasticsearch、Logstash、Kibana)等服务以进行集中式日志管理。这些工具提供强大的搜索和过滤选项,可以快速解决生产问题。

  • 性能注意事项:
    日志记录会影响性能,因此请避免过多的日志记录,尤其是在高流量应用程序中。考虑为开发(调试)和生产(信息或错误)设置不同的日志级别。

  • 保护日志:
    为防止敏感信息泄露,请避免记录敏感数据,或使用脱敏技术对用户信息进行匿名化。

6。现实世界的例子

  • 电子商务应用程序:对于电子商务应用程序,记录用户操作并监控结帐过程中的错误可以深入了解用户行为和阻碍转化的问题。
  • 具有结构化日志记录的 REST API:使用结构化日志记录的 REST API 可以跟踪请求详细信息、错误和用户访问,从而帮助维护系统可靠性。

结论

总之,Node.js 中设计良好的错误处理和日志记录策略对于构建有弹性且可维护的应用程序至关重要。有效的错误处理和结构化日志记录使开发人员能够监视、排除故障和优化应用程序,确保他们准备好妥善处理错误并跟踪应用程序的运行状况。通过将这些技术与持续监控相结合,您将拥有一个更可靠的系统可供生产。

版本声明 本文转载于:https://dev.to/divine_nnanna2/error-handling-and-logging-in-nodejs-applications-1k2a?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • PHP 中可以不继承地修改类方法吗?
    PHP 中可以不继承地修改类方法吗?
    我可以在没有继承的情况下对类进行猴子修补吗?您可能会遇到需要修改类或其方法而没有典型继承选项的情况。以下面的类为例:class third_party_library { function buggy_function() { return 'bad result'; ...
    编程 发布于2024-11-17
  • 如何使用纯 JavaScript 自动调整文本区域大小?
    如何使用纯 JavaScript 自动调整文本区域大小?
    Textarea 自动高度这个问题旨在消除文本区域的滚动条并调整其高度以匹配其中的内容。提供了使用纯JavaScript代码的解决方案:function auto_grow(element) { element.style.height = "5px"; element....
    编程 发布于2024-11-17
  • 什么时候手动调用析构函数是合法的做法?
    什么时候手动调用析构函数是合法的做法?
    什么时候手动调用析构函数是合理的?手动调用析构函数表明存在缺陷的设计这一概念经常被断言。然而,这引出了一个问题:这条规则有例外吗?反例:需要手动析构函数调用的情况确实,在某些情况下有必要这样做显式调用析构函数:1.受控内存释放:当内存分配和释放独立于对象构造和销毁进行管理时,手动析构函数调用变得至关...
    编程 发布于2024-11-17
  • 大批
    大批
    方法是可以在对象上调用的 fns 数组是对象,因此它们在 JS 中也有方法。 slice(begin):将数组的一部分提取到新数组中,而不改变原始数组。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index p...
    编程 发布于2024-11-17
  • 如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    如何在 PHP 中组合两个关联数组,同时保留唯一 ID 并处理重复名称?
    在 PHP 中组合关联数组在 PHP 中,将两个关联数组组合成一个数组是一项常见任务。考虑以下请求:问题描述:提供的代码定义了两个关联数组,$array1 和 $array2。目标是创建一个新数组 $array3,它合并两个数组中的所有键值对。 此外,提供的数组具有唯一的 ID,而名称可能重合。要求...
    编程 发布于2024-11-17
  • 除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    除了“if”语句之外:还有哪些地方可以在不进行强制转换的情况下使用具有显式“bool”转换的类型?
    无需强制转换即可上下文转换为 bool您的类定义了对 bool 的显式转换,使您能够在条件语句中直接使用其实例“t”。然而,这种显式转换提出了一个问题:“t”在哪里可以在不进行强制转换的情况下用作 bool?上下文转换场景C 标准指定了四种值可以根据上下文转换为的主要场景bool:语句:if、whi...
    编程 发布于2024-11-17
  • 使用 Go Module 部署 Go Cloud Functions 时如何解决“Build failed: go: parsing /models/go.mod: open /models/go.mod: no such file or directory\”错误?
    使用 Go Module 部署 Go Cloud Functions 时如何解决“Build failed: go: parsing /models/go.mod: open /models/go.mod: no such file or directory\”错误?
    使用 Go 模块部署 Google Cloud Function 错误尝试使用 Go 模块在 Go 1.11 中部署 Google Cloud Function 时,开发者可能会遇到错误,“构建失败:go:解析/models/go.mod:打开/models/go.mod:没有这样的文件或目录。&q...
    编程 发布于2024-11-17
  • 为什么我的 Golang `exec.Command` 返回“退出状态 1”?
    为什么我的 Golang `exec.Command` 返回“退出状态 1”?
    如何查明Golang的exec.Command中出现“Exit Status 1”错误的原因在Golang中执行exec.Command方法时,收到“退出状态 1”错误可能会非常模糊。缺乏特定信息会阻碍有效的调试。要检索更详细的信息,请利用 Command 对象的 Stderr 属性。这是通过以下方...
    编程 发布于2024-11-17
  • 如何使用“setState”更新状态中的嵌套对象?
    如何使用“setState”更新状态中的嵌套对象?
    使用 setState 更新 state.item[1] 在这种情况下,您正在尝试更新对象内的对象使用 setState 在您的状态中。要正确更新状态,您需要复制整个状态对象,修改副本,然后将状态设置为更新后的副本。以下是如何更新 state.item[1] 使用setState:// 1. Mak...
    编程 发布于2024-11-17
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-11-17
  • 将 PHP 连接到数据库:MySQL 初学者
    将 PHP 连接到数据库:MySQL 初学者
    在 PHP 中连接到 MySQL 数据库至关重要,步骤如下:安装 MySQL 扩展使用 mysqli_connect() 函数创建连接,参数包括主机、用户名、密码和数据库名称使用 mysqli_connect_error() 函数检查连接是否成功实战案例:获取所有用户,通过查询数据库并使用 mysq...
    编程 发布于2024-11-17
  • 为什么 CSS `visibility:hidden` 无法实现悬停效果?
    为什么 CSS `visibility:hidden` 无法实现悬停效果?
    揭开谜团:为什么 CSS 可见性在悬停时失败CSS 可见性提供了一种操作元素可见性的便捷方法,但有时它会偶然发现意想不到的障碍。考虑这样一个场景,您定义了一个“扰流器”类来使文本最初不可见,并在鼠标悬停时显示它。尽管您有期望,文本仍然顽固地保持在隐藏状态,无视您的悬停努力。深入探讨原因这种令人困惑的...
    编程 发布于2024-11-17
  • 何时在 JavaScript 中使用 parseInt() 和 Number() 进行字符串到数字的转换?
    何时在 JavaScript 中使用 parseInt() 和 Number() 进行字符串到数字的转换?
    使用parseInt()和Number()将字符串转换为数字在JavaScript中将字符串转换为数字时,两个常用的函数是parseInt () 和数字 ()。虽然它们都具有相同的数值转换目的,但它们的方法和行为有所不同。parseInt()parseInt() 执行更具体的任务,称为解析。它尝试从...
    编程 发布于2024-11-17
  • 项目 避免不必要地使用检查异常
    项目 避免不必要地使用检查异常
    检查异常是 Java 中的一个强大工具,因为它们迫使程序员处理异常情况,从而提高代码可靠性。然而,过度使用可能会导致 API 难以使用。为了证明检查异常是合理的,情况必须是真正的异常并且程序员能够采取有用的操作。否则,未经检查的异常可能更合适。 Java 8 给检查异常的使用带来了额外的挑战,因为抛...
    编程 发布于2024-11-17
  • 如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 查找今天生日的用户?
    如何使用 MySQL 识别今天生日的用户使用 MySQL 确定今天是否是用户的生日涉及查找生日匹配的所有行今天的日期。这可以通过一个简单的 MySQL 查询来实现,该查询将存储为 UNIX 时间戳的生日与今天的日期进行比较。以下 SQL 查询将获取今天有生日的所有用户: FROM USERS ...
    编程 发布于2024-11-17

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

Copyright© 2022 湘ICP备2022001581号-3