站点示例:
import vibe.d;
void login(HTTPServerRequest req, HTTPServerResponse res)
{
enforceHTTP("username" in req.form && "password" in req.form,
HTTPStatus.badRequest, "Missing username/password field.");
// todo: verify user/password here
auto session = res.startSession();
session["username"] = req.form["username"];
session["password"] = req.form["password"];
res.redirect("/home");
}
void logout(HTTPServerRequest req, HTTPServerResponse res)
{
res.terminateSession();
res.redirect("/");
}
void checkLogin(HTTPServerRequest req, HTTPServerResponse res)
{
// force a redirect to / for unauthenticated users
if( req.session is null )
res.redirect("/");
}
shared static this()
{
auto router = new URLRouter;
router.get("/", staticTemplate!"index.dl");
router.post("/login", &login);
router.post("/logout", &logout);
// restrict all following routes to authenticated users:
router.any("*", &checkLogin);
router.get("/home", staticTemplate!"home.dl");
auto settings = new HTTPServerSettings;
settings.sessionStore = new MemorySessionStore;
// ...
}
但我不想在整个程序中将ServerResponse传递给每个函数。例如,如果res.session存储的是当前用户的id,该怎么办。这是经常使用的,所以我不希望它通过每个函数。如何在全球范围内存储此会话信息?假设有多个用户在使用该网站。
这是可能的,尽管有点气馁。
解决方案
您不能简单地全局存储它,因为不存在"全局"会话这样的东西——它总是特定于请求上下文。由于多个光纤共享同一个执行线程,即使在D中默认将全局线程设置为本地线程也无济于事。
对于这样的任务,vibe.d提供了TaskLocal模板,它允许你做你想做的事情。我还没有尝试过,但希望TaskLocal!Session session
能"正常工作"。
请注意文档中的这两个警告:
但是,请注意,每个TaskLocal变量都会增加使用任务本地存储的任何任务的内存占用。访问TaskLocal变量也有开销,高于线程本地变量,但通常仍为O(1)
和
FiberLocal实例必须声明为静态/全局线程本地变量。将它们定义为临时/堆栈变量将导致崩溃或数据损坏!
反对
然而,根据我的经验,尽管打字有一些不便,但最好还是明确地传递所有这些上下文。不鼓励使用全局变量是有原因的——随着程序规模的增长,不可避免地很难跟踪模块之间的依赖关系并测试此类代码。现在陷入便利的诱惑可能会在以后引起很多头痛。为了尽量减少额外的类型并简化代码维护,我可能建议定义struct Context { ...}
,它将包含请求/会话指针,并将定期传递。