对express中next函数的一些理解

<div class="topic_content"> <div class="markdown-text"><p>最近公司在使用node做前后端分离,采用的web框架是express,所以对express框架进行了深入的了解,前段时间写了篇关于express路由的文章,但是在那篇文章中貌似少了一个很重要的内容,就是express的next,所以今...
摘要由CSDN通过智能技术生成

关于next主要从三点来进行说明:

  1. next的作用是什么?
  2. 我们应该在何时使用next?
  3. next的内部实现机制是什么?

Next的作用

我们在定义express中间件函数的时候都会将第三个参数定义为next,这个next就是我们今天的主角,next函数主要负责将控制权交给下一个中间件,如果当前中间件没有终结请求,并且next没有被调用,那么请求将被挂起,后边定义的中间件将得不到被执行的机会。

何时使用Next

从上边的描述我们已经知道,next函数主要是用来确保所有注册的中间件被一个接一个的执行,那么我们就应该在所有的中间件中调用next函数,但有一个特例,如果我们定义的中间件终结了本次请求,那就不应该再调用next函数,否则就可能会出问题,我们来看段代码

发送请求"/a",控制台打印日志如下:

404
GET /a 500 6.837 ms - -
Error: Can't set headers after they are sent.
    at ServerResponse.OutgoingMessage.setHeader (_http_outgoing.js:345:11)

为什么代码会抛异常呢,就是因为我们在res.send之后调用了next函数,虽然我们本次的请求已经被终止,但后边的404中间件依旧会被执行,而后边的中间件试图去向res的headers中添加属性值,所以就会抛出上边的异常。

读到这你可能会有个疑问,如果我不在res.send后边调用next函数,那后边定义的404中间件是不是永远都不会被执行到。现在我们删除res.send后边next函数调用,发送请求"/xxx",我们就会发现404中间件被执行了,(ㄒoㄒ),这不是和我们之前说的矛盾了吗,我们的自定义中间件没有调用next,但后边定义的中间件仍旧被执行了,这究竟是为什么呢。看来只能求助源码了~~~

Next的内部机制

function next(err) {
   
    ... //此处源码省略
    // find next matching layer
    var layer;
    var match;
    var route;
</span><span class="kwd">while</span><span class="pln"> </span><span class="pun">(</span><span class="pln">match </span><span class="pun">!==</span><span class="pln"> </span><span class="kwd">true</span><span class="pln"> </span><span class="pun">&amp;&amp;</span><span class="pln"> idx </span><span class="pun">&lt;</span><span class="pln"> stack</span><span class="pun">.</span><span class="pln">length</span><span class="pun">)</span><span class="pln"> </span><span class="pun">{</span><span class="pln">
  layer </span><span class="pun">=</span><span class="pln"> stack</span><span class="pun">[</span><span class="pln">idx</span><span class="pun">++];</span><span class="pln&
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 使用异步函数是不会逃脱 express app.use 函数的捕获错误机制的。 在 express ,app.use 函数用于添加间件函数,这些函数会在请求和响应的生命周期被调用。如果在这些函数抛出了错误,那么 express 会捕获这个错误并将其传递给错误处理间件。 因此,无论是同步函数还是异步函数,都不会逃脱 express app.use 函数的捕获错误机制。如果你在 app.use 函数使用了异步函数,那么只需要在函数内部使用 try-catch 语句或者在函数最后使用 await 来捕获并处理异步操作可能出现的错误即可。 ### 回答2: 在Express,通过app.use函数注册的间件会自动捕获发生在其内部的同步错误,并通过next函数将错误传递给错误处理间件进行处理。这种错误捕获机制可以有效地处理同步错误。然而,对于异步函数Express的错误处理机制是无法自动捕获并处理的。 使用异步函数时,如果在异步函数内部发生错误,Express的错误处理机制将无法捕获到这些错误,也无法进行相应的处理。这是因为异步函数的执行是非阻塞的,它们会立即返回一个Promise对象,并在后台执行。在这种情况下,错误发生在异步函数内部,而Express的错误处理机制在异步函数返回Promise对象后已经无法获取到这些错误。 为了解决这个问题,我们可以使用try/catch语句来捕获异步函数内部的错误,并手动调用next函数传递错误。这样,错误将被传递到下一个错误处理间件进行处理。 示例代码如下: app.use(async (req, res, next) => { try { // 异步操作 await someAsyncFunction(); next(); } catch (err) { next(err); } }); 上述代码,我们使用了async/await语法来处理异步函数,然后在try块执行异步操作,如果发生错误,则通过next函数将错误传递给下一个错误处理间件进行处理。 综上所述,使用异步函数时,需要手动处理其发生的错误,通过try/catch语句捕获错误并调用next函数传递错误,以便Express能够正确地处理它们。 ### 回答3: 使用异步函数不会逃脱express`app.use`函数的捕获错误的机制。 在Express,通过定义间件函数来处理HTTP请求,这些间件函数可以是同步的,也可以是异步的。无论是同步还是异步的间件函数Express都会在调用它们时使用try-catch块来捕获可能出现的错误。 在异步函数,当出现错误时,如果没有使用try-catch语句来显式地捕获错误,这些错误将会被自动转发到错误处理间件。错误处理间件是一个专门用来处理错误的间件函数,可以用`app.use`或`app.use(errorHandler)`方式来定义。因此,无论是同步的还是异步的函数,只要在处理请求的过程出现错误,Express都会通过错误处理间件来捕获和处理这些错误。 例如,下面是一个使用异步函数Express间件的示例: ``` app.use(async (req, res, next) => { try { // 异步操作 await someAsyncFunction(); // 继续处理请求 next(); } catch (err) { // 错误处理 next(err); } }); ``` 在上述示例,`someAsyncFunction`是一个异步函数,当它抛出一个错误时,try-catch块会捕获这个错误并通过`next(err)`将其传递给错误处理间件进行处理。 总之,无论使用同步还是异步函数Express都会捕获这些函数可能抛出的错误,并通过错误处理间件来处理这些错误,使得我们能够更好地处理和管理错误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值