错误:类型参数“D”必须用作某些本地类型的类型参数



我正在使用MongoDB的 Nickel.rs 来构建RESTful api。我想为mongodb::error::Result<Option<bson::Document>>类型实现一个通用Responder

这是我根据我为Responder找到的示例编写的实现:

impl<D> Responder<D> for Result<Option<Document>> {
    fn respond<'a>(self, mut response: Response<'a, D>) -> MiddlewareResult<'a, D> {
        response.set(MediaType::Json);
        match self {
            Ok(Some(doc))=>{
                ApiResponse{data: Bson::Document(doc).to_json()}.to_json()
            },
            Ok(None)=>{
                response.set(StatusCode::NotFound);
                ApiError{error: "Not found".to_string()}.to_json()
            },
            Err(e)=>{
                response.set(StatusCode::InternalServerError);
                ApiError{error: format!("{}",e)}.to_json()
            }
        }
    }
}

并且我收到以下错误:

错误:类型参数D必须用作某些类型的参数 本地类型(例如 MyStruct<T> );仅在当前定义的特征 可以为类型参数实现 crate [E0210]

我跑rustc --explain E0210解释,如果我的理解是正确的,我需要提供一个特征D作为impl<D>的类型参数,但我不明白提供哪个特征。

我尝试了impl<D: =()>但产生了相同的错误。

当你实现一个特征时,必须在同一条箱中定义这个特征或你实现它的类型。 在您的示例中,情况并非如此:特征Respondernickel定义,而Resultmongodb定义。

解决此问题的常见方法是定义自己的类型,方法是将所需类型包装到具有单个组件的元组结构中(所谓的 newtype 模式):

struct Result(mongodb::error::Result<Option<Document>>);
impl Responder for Result {
    ...

根据 starblue 的回答,我用元组结构替换了 ApiResponseApiError,并重构了我的代码,如下所示:

struct ApiResponse<T>(T);
impl<D> Responder<D> for ApiResponse<Result<Option<Document>>> {
    fn respond<'a>(self, mut response: Response<'a, D>) -> MiddlewareResult<'a, D> {
        let mut d = BTreeMap::new();
        match self.0 {
            Ok(Some(doc))=>{
                d.insert("data".to_string(),Bson::Document(doc).to_json());
            },
            Ok(None)=>{
                response.set(StatusCode::NotFound);
                d.insert("error".to_string(),"Not Found".to_json());
            },
            Err(e)=>{
                response.set(StatusCode::InternalServerError);
                d.insert("error".to_string(),format!("{}",e).to_json());
            }
        }
        response.set(MediaType::Json);
        response.send(Json::Object(d))
    }
}

最新更新