【PhysUnits】13 减法操作(sub.rs)

一、源码

这段代码定义了一个自定义的减法 trait MySub 并为实现了特定约束的整数类型实现了这个 trait。

rust 复制代码
use super::basic::Integer;
use core::ops::{Neg, Sub, Add};

pub trait MySub<Rhs = Self> {
    type Output;
    fn my_sub(self, rhs: Rhs) -> Self::Output;
}

impl<I1: Integer, I2: Integer> MySub<I2> for I1
where
    I1: Add<<I2 as Neg>::Output>,
    I2: Neg,
{
    type Output = <I1 as Add<<I2 as Neg>::Output>>::Output;
    
    #[inline]
    fn my_sub(self, rhs: I2) -> Self::Output {
        self + (-rhs)
    }
}

二、导入部分

rust 复制代码
use super::basic::Integer;
use core::ops::{Neg, Sub, Add};
  • 从父模块的 basic 子模块导入 Integer trait

  • 从核心库导入 Neg(取负)、Sub(减法) 和 Add(加法) 这三个操作 trait

三、自定义减法 trait 定义

rust 复制代码
pub trait MySub<Rhs = Self> {
    type Output;
    fn my_sub(self, rhs: Rhs) -> Self::Output;
}
`
+ 定义了一个名为 MySub 的公共 trait

+ 使用泛型参数 Rhs (Right Hand Side 的缩写),默认值为 Self (即实现该 trait 的类型自身)

+ 声明了一个关联类型 Output 表示运算结果的类型

+ 定义了一个方法 my_sub,接收自身和右操作数,返回 Output 类型的结果

### 四、trait 实现
```rust
impl<I1: Integer, I2: Integer> MySub<I2> for I1
where
    I1: Add<<I2 as Neg>::Output>,
    I2: Neg,
{
    type Output = <I1 as Add<<I2 as Neg>::Output>>::Output;
    
    #[inline]
    fn my_sub(self, rhs: I2) -> Self::Output {
        self + (-rhs)
    }
}
  • 为所有实现了 Integer 的类型 I1 和 I2 实现 MySub

  • where 子句约束:

    • I1 必须能加上 I2 的负值 (I1: Add<::Output>)

    • I2 必须能取负 (I2: Neg)

  • 关联类型 Output 定义为 I1 加 I2 的负值的结果类型

  • my_sub 方法实现为 self + (-rhs),即通过加负来实现减

  • #[inline] 提示编译器考虑内联此方法以提高性能

五、关键点解析

  1. 数学原理:
  • 这段代码实现了减法作为加负数的操作 (a - b = a + (-b))

  • 这是数学上减法的标准定义方式

  1. 类型系统:
  • 使用了复杂的类型约束确保类型安全

  • I2 as Neg>::Output 获取 I2 取负后的类型

  • I1 as Add<...>::Output 获取 I1 与某类型相加的结果类型

  1. 设计考虑:
  • 自定义 MySub trait 是为了绕过 Rust 的孤儿规则

  • 保持了与标准库 Sub trait 相似的接口设计

六、使用示例

假设有以下类型:

rust 复制代码
struct MyInt(i32);

impl Integer for MyInt {...}
impl Neg for MyInt {
    type Output = MyInt;
    fn neg(self) -> Self::Output {...}
}
impl Add for MyInt {
    type Output = MyInt;
    fn add(self, rhs: Self) -> Self::Output {...}
}

可以这样使用:

rust 复制代码
let a = MyInt(5);
let b = MyInt(3);
let c = a.my_sub(b);  // 相当于 5 - 3

七、与标准库 Sub 的区别

  1. 避免了孤儿规则问题,因为 MySub 是本地的

  2. 方法名改为 my_sub 而非 sub

  3. 其他方面保持了与标准库 Sub 相同的设计理念

这种模式在需要扩展外部类型功能时很常见,特别是在无法直接实现外部 crate 的 trait 时。

相关推荐
hdsoft_huge5 小时前
VUE npm ERR! code ERESOLVE, npm ERR! ERESOLVE could not resolve, 错误有效解决
vue.js·rust·npm
无名之逆1 天前
WebSocket Broadcasting with hyperlane
网络·websocket·网络协议·http·rust
刃神太酷啦2 天前
类和对象(5)--《Hello C++ Wrold!》(7)--(C/C++)--构造函数的初始化列表,explicit关键词,友元,内部类和匿名对象
java·c语言·c++·算法·rust·动态规划
攻城狮7号2 天前
Rust 1.0 发布十周年,梦想再度扬帆起航!
开发语言·后端·游戏·rust
芳草萋萋鹦鹉洲哦2 天前
【Macos】安装前端环境rust+node环境
前端·macos·rust
后青春期的诗go2 天前
基于Rust语言的Rocket框架和Sqlx库开发WebAPI项目记录(五)
开发语言·后端·rust·rocket
Source.Liu4 天前
【PhysUnits】10 减一操作(sub1.rs)
rust
明月看潮生4 天前
青少年编程与数学 02-019 Rust 编程基础 23课题、web服务器
服务器·开发语言·青少年编程·rust
UestcXiye5 天前
Rust 学习笔记:泛型
rust