有没有办法让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();
}
首先,创建一个结构,它是应用程序的共享状态:
struct AppState {
my_big_heavy_object: HeavyObject,
}
最好创建这样的上下文包装器,而不是仅使用HeavyObject
,以便您可以稍后在必要时向其添加其他字段。
Actix 中的一些对象现在需要与此对象交互,因此您可以通过覆盖这些类型中的应用程序状态参数(例如HttpRequest<AppState>
(来使它们意识到这一点。
您的index
处理程序可以通过HttpRequest
的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
或RefCell
之类的东西来实现内部可变性。