一尘不染

Node.js / Express.js-app.router如何工作?

node.js

在我提出app.router疑问之前,我至少应该解释一下使用中间件时发生的事情。要使用中间件,要使用的功能是app.use()。在执行中间件时,它将使用next()或使其调用下一个中间件,从而不再调用任何中间件。这意味着我放置中间件调用的顺序很重要,因为某些中间件依赖于其他中间件,而接近末尾的某些中间件甚至可能不会被调用。

今天,我正在开发应用程序,并在后台运行服务器。我想进行一些更改并刷新页面,然后立即查看更改。具体来说,我正在更改布局。我无法使它正常工作,所以我在Stack
Overflow中搜索了答案,并找到了这个问题。它说要确保它express.static()在下面require('stylus')。但是,当我查看该OP的代码时,我发现他app.router在中间件调用的最后就接到了他的电话,而我试图弄清楚为什么会这样。

当我制作Express.js应用程序(版本3.0.0rc4)时,我使用了命令,express app --sessions --css stylus并且在我的app.js文件中,app.routerexpress.static()require('stylus')调用上方都设置了代码。如此看来,如果它已经以这种方式设置,那么它应该保持这种方式。

重新排列代码以便可以看到手写笔更改后,它看起来像这样:

app.configure(function(){
  //app.set() calls
  //app.use() calls
  //...
  app.use(app.router);
  app.use(require('stylus').middleware(__dirname + '/public'));
  app.use(express.static(__dirname + '/public', {maxAge: 31557600000}));
});

app.get('/', routes.index);

app.get('/test', function(req, res){
  res.send('Test');
});

因此,我决定第一步就是要弄清楚为什么甚至app.router在我的代码中也是如此。因此我将其注释掉,启动了我的应用程序并导航至/。它显示我的索引页面很好。嗯,也许是可行的,因为我正在从我的路由文件(routes.index)中导出路由。因此,接下来我导航到/test它,并在屏幕上显示Test。哈哈,好吧,我不知道该怎么办app.router。无论它是否包含在我的代码中,我的路由都可以。因此,我肯定会缺少一些东西。

所以这是我的问题:

有人可以解释一下app.router它的作用,重要性以及在中间件调用中的位置吗?如果得到有关的简短说明,那也很好express.static()。据我所知,express.static()是我的信息的缓存,如果应用程序找不到请求的页面,它将检查缓存以查看其是否存在。


阅读 278

收藏
2020-07-07

共1个答案

一尘不染

注意: 这描述了Express在版本2和版本3中的工作方式。有关Express 4的信息,请参阅本文结尾。


static只需提供磁盘中的文件( 静态 资源)即可。您给它提供一个路径(有时称为挂载点),它为该文件夹中的文件提供服务。

例如,express.static('/var/www')将提供该文件夹中的文件。所以到您的节点服务器的请求,http://server/file.html将有助于/var/www/file.html

router是运行您的路线的代码。当您执行时app.get('/user', function(req, res) { ... });router实际上是调用回调函数来处理请求。

您将事物传递app.use的顺序确定了使每个中间件有机会处理请求的顺序。例如,如果您test.html的静态文件夹中有一个名为的文件和一条路由:

app.get('/test.html', function(req, res) {
    res.send('Hello from route handler');
});

哪一个发送给客户请求http://server/test.html?首先使用哪种中间件use

如果您这样做:

app.use(express.static(__dirname + '/public'));
app.use(app.router);

然后,将提供磁盘上的文件。

如果你反过来做,

app.use(app.router);
app.use(express.static(__dirname + '/public'));

然后,路由处理程序获取请求,并且“来自路由处理程序的Hello”将发送到浏览器。

通常,您希望将路由器放置在静态中间件 上方 ,以免意外命名的文件无法覆盖您的路由之一。

请注意,如果你不明确userouter,它是隐含快递在您定义的路线点添加(这就是为什么你的路线仍然工作,即使你注释掉app.use(app.router))。


评论者提出了关于顺序的另一点,staticrouter我尚未解决:对应用程序整体性能的影响。

use
router上面的另一个原因static是优化性能。如果您放static第一把,那么您将在每个单独的请求上访问硬盘驱动器,以查看文件是否存在。在快速测试中,我发现在卸载的服务器上,此开销总计约为1ms。(该数字很可能在负载下更高,因为请求将竞争磁盘访问。)

有了router第一次,从来没有匹配的路由请求有打盘,节省了宝贵的毫秒。

当然,有一些方法可以减轻static的开销。

最好的选择是将所有静态资源放在特定的文件夹下。(IE
/static)然后static,您可以安装到该路径,以便仅在路径开头为时才运行/static

app.use('/static', express.static(__dirname + '/static'));

在这种情况下,您可以将其放在上方router。如果存在文件,这可以避免处理其他中间件/路由器,但是老实说,我怀疑您会获得这么多。

您还可以使用staticCache,它在内存中缓存静态资源,这样您就不必在磁盘上存放常用文件。( 警告: staticCache显然将来会被删除。)

但是,我认为不staticCache缓存否定答案(当文件不存在时),因此如果您未将其挂载到路径staticCacherouter而放在上面,这将无济于事。

与所有有关性能的问题一样 ,请对您的实际应用 (在负载下)进行 测量并进行基准测试, 以查看瓶颈所在。


快递4

Express 4.0 删除
app.router。现在,所有中间件(app.use)和路由(app.get等)都按照添加它们的精确顺序进行处理。

换一种说法:

所有路由方法将按照它们出现的顺序添加。你不应该 这样app.use(app.router)。这样可以消除Express中最常见的问题。

换句话说,混合app.use()app[VERB]()完全 按照它们被调用的顺序工作。

app.get('/', home);
app.use('/public', require('st')(process.cwd()));
app.get('/users', users.list);
app.post('/users', users.create);

阅读有关Express 4中更改的更多信息。

2020-07-07