AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79249127
Accepted
Josh Burkart
Josh Burkart
Asked: 2024-12-04 04:46:18 +0800 CST2024-12-04 04:46:18 +0800 CST 2024-12-04 04:46:18 +0800 CST

定义并全面实现一个共同的特征以避免 Rust 中的依赖关系

  • 772

高级问题陈述

我使用一个通用的声明性宏在许多不同的 crate 中定义了各种通用容器。每个容器看起来都像这样:

struct Container1<T> {
    a: T,
    b: T,
}

另一方面,我有一个单独的板条箱model,它具有一个特征ModelTree,当它们的类型参数实现其他特征时,我想在所有通用容器上实现该特征Model:

trait ModelTree {
    fn integrate_model_tree(self, context: Context);
}

理想情况下应该是

impl<T: Model> ModelTree for Container1<T> {
    ...
}

[...and similarly for other containers...]

但是,该model包不能直接依赖于定义容器的所有包(有很多),并且定义容器的包也不能依赖于该model包。

为了解决这个问题,我考虑从高层次上定义一个通用的 crate common,它定义了一个“类似于”的特性ModelTree,让所有容器(例如Container1)实现这个通用特性,然后ModelTree对实现这个通用特性的任何东西进行通用实现。然而,unconstrained type parameter每当我尝试这样做时,我都会不断得到错误。

具体尝试

以下是我在 中尝试的具体操作common。此游乐场链接包含完整的 MRE。定义一个常见的访客特征:

trait Visitor {
    type ItemType;

    fn visit(&mut self, item: Self::ItemType);
}

定义“可访问”的特征AcceptVisitor:

trait AcceptVisitor<V: Visitor<ItemType = Self::ItemType>> {
    type ItemType;

    fn accept_visitor(self, visitor: V);
}

AcceptVisitor为所有通用容器实施:

impl<T, V: Visitor<ItemType = T>> AcceptVisitor<V> for Container1<T> {
    type ItemType = T;

    fn accept_visitor(self, mut visitor: V) {
        visitor.visit(self.a);
        visitor.visit(self.b);
    }
}

在箱子中添加访客垫片model:

struct ModelVisitor<M: Model> {
    context: Context,
    phantom: core::marker::PhantomData<M>,
}

impl<M: Model> Visitor for ModelVisitor<M> {
    type ItemType = M;
    
    fn visit(&mut self, item: M) {
        self.context.add_model(item);
    }
}

最后,ModelTree针对所有实现的类型实现AcceptVisitor:

impl<
        M: Model,
        V: Visitor<ItemType = M>,
        A: AcceptVisitor<ModelVisitor<M>, ItemType = M>,
    > ModelTree for A
{
    fn integrate_model_tree(self, context: Context) {
        let visitor = ModelVisitor { context, phantom: core::marker::PhantomData };
        self.accept_visitor(visitor);
    }
}

这就是我得到和的unconstrained type parameter错误的地方。MV

问题

有什么方法可以实现我的高级问题陈述?我的具体尝试是否走错了路,还是有办法对其进行调整以使其发挥作用?谢谢!

generics
  • 1 1 个回答
  • 22 Views

1 个回答

  • Voted
  1. Best Answer
    Chayim Friedman
    2024-12-04T05:21:56+08:002024-12-04T05:21:56+08:00

    这听起来很像XY 问题,但如果这确实是你想要做的,你可以将Visitor泛型参数从特征移到accept_visitor()方法。这也是更正确的,因为实现不依赖于单个访问者。

    trait AcceptVisitor {
        type ItemType;
    
        fn accept_visitor<V: Visitor<ItemType = Self::ItemType>>(self, visitor: V);
    }
    
    impl<T> AcceptVisitor for Container1<T> {
        type ItemType = T;
    
        fn accept_visitor<V: Visitor<ItemType = T>>(self, mut visitor: V) {
            visitor.visit(self.a);
            visitor.visit(self.b);
        }
    }
    
    impl<M: Model, A: AcceptVisitor<ItemType = M>> ModelTree for A {
        fn integrate_model_tree(self, context: Context) {
            let visitor = ModelVisitor {
                context,
                phantom: core::marker::PhantomData,
            };
            self.accept_visitor(visitor);
        }
    }
    
    • 2

相关问题

  • Rust 无法推断 Fn 类型,其中 Option<Fn> 为 None [重复]

  • 如何在 Rust 中传递一个以迭代器作为函数参数的函数?

  • Rust Specs Crate 系统中的通用组件

  • 将一种泛型类型转换为另一种泛型类型

  • 有没有办法扩展依赖于 F# 中的 K 和 V 的泛型类型 IStore<ValueTuple<K, V>> ?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve