有没有办法让actix-web
路由处理程序知道计算result
所需的预先计算的重型对象?
最终,我打算做的是避免每次请求出现时重新计算my_big_heavy_object
,而是在main
中一次性计算,然后从index
方法。
extern crate actix_web;
use actix_web::{server, App, HttpRequest};
fn index(_req: HttpRequest) -> HttpResponse {
// how can I access my_big_heavy_object from here?
let result = do_something_with(my_big_heavy_object);
HttpResponse::Ok()
.content_type("application/json")
.body(result)
}
fn main() {
let my_big_heavy_object = compute_big_heavy_object();
server::new(|| App::new().resource("/", |r| r.f(index)))
.bind("127.0.0.1:8088")
.unwrap()
.run();
}
答案 0 :(得分:1)
首先,创建一个结构,它是应用程序的共享状态:
struct AppState {
my_big_heavy_object: HeavyObject,
}
最好像这样创建一个上下文包装器,而不是仅使用HeavyObject
,这样您可以在以后必要时添加其他字段。
Actix中的一些对象现在需要与此进行交互,因此您可以通过覆盖这些类型中的应用程序状态参数来使其了解它,例如HttpRequest<AppState>
。
您的index
处理程序可以通过HttpRequest
的{{1}}属性访问该状态,现在看起来像这样:
state
构建fn index(req: HttpRequest<AppState>) -> HttpResponse {
let result = do_something_with(req.state.my_big_heavy_object);
HttpResponse::Ok()
.content_type("application/json")
.body(result)
}
时,请使用App
构造函数而不是with_state
:
new
请注意,假定应用程序状态是不可变的。听起来你不需要任何处理程序来改变它,但如果你这样做,你将不得不使用像server::new(|| {
let app_state = AppState { my_big_heavy_object: compute_big_heavy_object() };
App::with_state(app_state).resource("/", |r| r.f(index))
})
.bind("127.0.0.1:8088")
.unwrap()
.run();
或Cell
那样的内部可变性。