我从NodeJS和Express 4开始,我有点困惑。我一直在阅读Express网站,但不知道什么时候使用路由处理程序或什么时候使用Express . router。

正如我所看到的,如果我想在用户点击/show时显示一个页面或其他东西,例如,我应该使用:

var express = require('express')    
var app = express()    
app.get("/show", someFunction)  

一开始,我认为这是旧的(快车3)。这是对的,还是这是快车4的方式?

如果这是特快4的做法,什么是特快。路由器用于?

我看到了几乎和上面相同的例子,但是使用了express。路由器:

var express = require('express');
var router = express.Router();
router.get("/show", someFunction)

那么,这两个例子有什么不同呢?

如果我只是想做一个简单的测试网站,我应该使用哪一个?


当前回答

表达。路由器有很多选项:

启用区分大小写:/show route不与/show相同,默认情况下不启用此行为 严格路由模式:/show/路由到与/show不相同,此行为默认也被禁用 我们可以为特定的路由添加特定的中间件

其他回答

表达。路由器有很多选项:

启用区分大小写:/show route不与/show相同,默认情况下不启用此行为 严格路由模式:/show/路由到与/show不相同,此行为默认也被禁用 我们可以为特定的路由添加特定的中间件

在测试中有一个问题是这样问的:“express.Router()创建了一个行为类似于app对象的对象。”

正确答案是“True”。我知道我们都可以通过使用两者中的任何一个来创建路由器,但是否可以安全地说它们在所有情况下都不相同?如果我的理解是正确的,express()变量可以做更多的事情,比如启动服务器,而另一个变量不能。

它们有何不同

每个人,包括文档,都倾向于回顾它们有多少相同之处,但实际上并没有提及任何差异。事实上,它们是不同的。

var bigApp = express();
var miniApp = express.Router();

听()

最明显的区别是bigApp将提供listen,这只是一种相当令人困惑的方式来完成原本简单而明显的节点http或https模块:

var server = require('http').createServer(bigApp);

server.listen(8080, function () {
  console.info(server.address());  
});

我认为这是一种反模式,因为它抽象和模糊了一些本来并不复杂或困难的东西,然后让人们很难使用websockets和其他需要原始http服务器的中间件。

内部状态

最大的区别,也是非常重要的区别是,所有bigApps都有单独的内部状态。

bigApp.enable('trust proxy');
bigApp.enabled('trust proxy');
// true

var bigApp2 = express();
bigApp2.enabled('trust proxy');
// false

bigApp.use('/bunnies', bigApp2);
// WRONG! '/bunnies' will NOT trust proxies

然而,传递给bigApp的miniApp将由bigApp以这样一种方式进行操作,即它的内部状态和thisness将被保留,并且这些路由将相应地执行。

bigApp.enable('trust proxy');
bigApp.enabled('trust proxy');
// true

var miniApp = express.Router();

bigApp.use('/bunnies', miniApp);
// CORRECT! All state and such are preserved

这可能是一个大问题,因为express对http做了很多(有时是三重的)事情。ServerRequest和httpServerResponse对象——比如修改(或劫持)req。Url和req。originalUrl和其他您一直在使用而没有意识到的属性-您可能不希望复制和分离它们。

小火

路由器可以使用的函数数量更少,定义更明确:

用(山,fn)。 所有(山,fn)。 选项(山,fn)。 头山(fn) 得到(山,fn)。 波斯特(山,fn)。 (山贴片,fn)。 普特(山,fn)。 删除(山,fn)。 XXXX(山)公路。 帕拉姆(name, cb)。XXXX

还有其他一些方便的方法,比如basic(),但你找不到set()或enable()或其他方法来改变更大的应用程序状态。

app.js

var express = require('express'),
    dogs    = require('./routes/dogs'),
    cats    = require('./routes/cats'),
    birds   = require('./routes/birds');

var app = express();

app.use('/dogs',  dogs);
app.use('/cats',  cats);
app.use('/birds', birds);

app.listen(3000);

dogs.js

var express = require('express');

var router = express.Router();

router.get('/', function(req, res) {
    res.send('GET handler for /dogs route.');
});

router.post('/', function(req, res) {
    res.send('POST handler for /dogs route.');
});

module.exports = router;

当var app = express()被调用时,返回一个app对象。把这个看作是主应用。

当var router = express.Router()被调用时,返回一个略有不同的小应用程序。迷你应用程序背后的想法是,应用程序中的每条路由都可能变得相当复杂,将所有代码移到一个单独的文件中会受益。每个文件的路由器都变成了一个小应用程序,它的结构与主应用程序非常相似。

在上面的例子中,/dogs路径的代码已经被移动到它自己的文件中,所以它不会把主应用程序弄得乱七八糟。/cats和/birds的代码在它们自己的文件中也会有类似的结构。通过将此代码分离为三个小应用程序,您可以单独处理每个小应用程序的逻辑,而不必担心它将如何影响其他两个小应用程序。

如果你有属于这三种路由的代码(中间件),你可以在app.use(…)调用之前把它放在主应用程序中。如果您有只属于其中一条路由的代码(中间件),那么您可以将它只放在该路由的文件中。

假设您的应用程序有点复杂。因此,我们首先要做的是将应用程序划分为多个模块,这样一个模块中的更改就不会影响其他模块,您可以继续在各个模块上工作,但最终您需要将所有内容集成到一个模块中,因为您正在构建单个应用程序。这就像我们有一个主应用程序和几个子应用程序,它们的父应用程序是主应用程序。 当我们创建父应用程序时,我们使用

const express = require("express");
const parent = express();

对于父应用程序,我们需要引入子应用程序。但由于子应用程序并不是完全不同的应用程序(因为它们运行在相同的上下文中- java术语),express提供了通过express的Router函数来实现的方法,这就是我们在每个子模块文件中所做的,并调用这样的子模块aboutme。

const express = require("express");
export const router = express.Router();

通过导出,我们使这个模块可供其他人使用,由于我们已经模块化了,我们需要通过节点的require函数使模块文件可用于父应用程序,就像任何其他第三方模块一样,父文件看起来像这样:

const express = require("express");
const parent = express();
const child = require("./aboutme");

在我们让这个子模块对父应用程序可用之后,我们需要告诉父应用程序什么时候使用这个子应用程序。假设当一个用户点击了关于我的路径,我们需要关于我的子应用程序来处理请求,我们使用Express的use方法来做到这一点:

parent.use("/aboutme", child);

在一个镜头中,父文件看起来是这样的:

const express = require("express");
const parent = express();
const child = require("./aboutme");

parent.use("/aboutme", child);

最重要的是,父进程可以启动子进程不能启动的服务器。希望这能说明问题。要了解更多信息,你可以查看源代码,这需要一些时间,但它给了你很多信息。