【问题标题】:Rust generics: Expected <T> found <Foo>Rust 泛型:预期的 <T> 找到了 <Foo>
【发布时间】:2015-07-18 11:57:19
【问题描述】:

我正在尝试使用泛型,但我没有很好地掌握该主题,我收到此错误:

error: mismatched types:
expected `book::mdbook::MDBook<R>`,
found `book::mdbook::MDBook<renderer::html_handlebars::HtmlHandlebars>`
(expected type parameter,
found struct `renderer::html_handlebars::HtmlHandlebars`) [E0308]

这是相关代码

pub struct MDBook<R> where R: Renderer {
    title: String,
    author: String,
    config: BookConfig,
    pub content: Vec<BookItem>,
    renderer: R,
}

impl<R> MDBook<R> where R: Renderer {

    pub fn new(path: &PathBuf) -> Self {

        MDBook {
            title: String::from(""),
            author: String::from(""),
            content: vec![],
            config: BookConfig::new()
                        .set_src(path.join("src"))
                        .set_dest(path.join("book")),
            renderer: HtmlHandlebars::new(), // <---- ERROR HERE
        }
    }
}

Renderer 特征目前为空,HtmlHandlebars 的实现为

pub struct HtmlHandlebars;

impl Renderer for HtmlHandlebars {

}

impl HtmlHandlebars {
    pub fn new() -> Self {
        HtmlHandlebars
    }
}

我做错了什么?

【问题讨论】:

  • 顺便说一句,&amp;PathBuf&amp;String 一样没用。你想要&amp;Path(道德上等同于&amp;str)。
  • 好的,谢谢!我不知道:)

标签: generics rust traits


【解决方案1】:
impl<R> MDBook<R> where R: Renderer {

    pub fn new(path: &PathBuf) -> Self {

这些行声称​​对于所有实现RendererR 类型,有一个方法new(path) 返回MDBook&lt;R&gt;。但是,无论R 是什么,您对该方法的实现总是返回MDBook&lt;HtmlHandlebars&gt;

您可以添加绑定到R 的特征(或绑定到Renderer 的方法),允许在new 中构造R 类型的值。或者,该方法可以接受渲染器作为参数,即fn new(path: &amp;Path, renderer: R) -&gt; Self。无论哪种方式,您都需要一种方法来获取new 内的渲染器(即R 类型的值)。

另一方面,如果你想支持这样的东西:

let book = MDBook::new(path);
if some_condition {
    book.set_renderer(SomeOtherThing::new());
}

那么泛型对于这项工作来说是错误的工具,因为它们使渲染器的选择成为book 静态类型的一部分。您可以完全删除R 类型参数,保留您的特征并简单地将trait object(可能是Box&lt;Renderer&gt;)存储在MDBook 中。

【讨论】:

  • 我希望new 始终返回MDBook&lt;HtmlHandlebars&gt;,因为它是默认渲染器。那么我应该将返回值从Self 更改为MDBook&lt;HtmlHandlebars&gt; 吗?
  • 实际上我想要实现的是能够将渲染器换成另一个。特质似乎是最好的方法......这是个好主意还是有更好的方法?
  • @MathieuDavid 您可以有一个返回MDBook&lt;HtmlHandlebars&gt; 的方法,尽管将它放入impl MDBook&lt;HtmlHandlebars&gt; {} 块中会更合适。但是,如果您没有其他方法可以与所有MDBook&lt;R&gt; 一起使用R,那么泛型就毫无意义。名称也必须不同,因为这里没有重载。也许问问自己为什么要为此默认设置特殊情况。
  • @MathieuDavid 如果您希望能够拥有一个MDBook 对象并动态更改它使用的渲染器,那么泛型是错误的工具(尽管 traits 对那个也是)。我也编辑了答案以涵盖这一点。
  • 非常感谢,使用Box&lt;Renderer&gt; 效果很好。
猜你喜欢
  • 2020-10-13
  • 2014-11-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多