当前位置: 首页 > news >正文

学习笔记二十——Rust trait

🧩 Rust Trait 彻底搞懂版

👀 目标读者:对 Rust 完全陌生,但想真正明白 “Trait、Trait Bound、孤岛法则” 在做什么、怎么用、为什么这样设计。
🛠 方法

  1. 先给“心里模型”——用生活类比把抽象概念掰开揉碎。
  2. 再给“最小代码”——跑得动、改得动,看编译器怎么说。
  3. 最后给“练习路线”——照着做,概念才能沉到肌肉里。

1️⃣ Trait 的心里模型——“技能证书”

现实类比Rust 中的名字说明
驾驶证:上面写可开货车/小客车Trait“会干什么”的清单,只列方法签名;没有数据
司机甲类型 (struct / enum)真正扛活儿的人
给司机颁证impl Trait for Type表示 “甲已掌握驾驶技能”

三点记住:

  1. Trait 不存数据,只规定行为。
  2. 一个类型可以拿多本证书 → 组合能力。
  3. 证书颁发 (impl) 时才写具体实现,编译期 就定好函数体,零额外开销。

2️⃣ 最小可跑例子

trait SayHi {                       // 证书:会打招呼fn hi(&self);                   // 方法清单:打招呼
}struct Cat { name: String }         // 司机:猫impl SayHi for Cat {                // 颁证:猫会打招呼fn hi(&self) {                  // 具体实现println!("喵,我是 {}", self.name);}
}fn main() {let kitty = Cat { name: "Tom".into() };kitty.hi();                     // 输出:喵,我是 Tom
}

3️⃣ Trait Bound 心里模型——“入场门票”

fn greet<T: SayHi>(v: &T) { v.hi(); }
  • 意思T 只有拿到 SayHi 证书 才能进场。
  • 写法扩展T: SayHi + Clone → 同时要两本证书;where 子句只是把字写到下一行更清爽。

4️⃣ PartialOrd + Copy 为何要一起写?

证书能力largest 为啥要它
PartialOrd能比较大小 (>, <)得知道谁更大
Copy能按位复制,不搬所有权返回最大值时不挪走原数据

组合写法 T: PartialOrd + Copy 就像门口贴“身高 1.6m 以上 年满 18 岁才能进”。


5️⃣ largest 函数剖面图(完全自定义名字)

fn largest<T: PartialOrd + Copy>(list: &[T]) -> T {let mut max = list[0];          // Copy 允许拷贝for &item in list {if item > max { max = item; } // PartialOrd 允许比较}max
}

练习:把 Copy 去掉再编译,看看错误提示,体会“证书缺了一本”的感觉。


6️⃣ impl Trait vs dyn Trait:两种“请师傅干活”的方式

问题impl Trait(静态,一对一)dyn Trait(动态,一群人)
具体类型编译期确定吗?✅ 是❌ 否(需到运行期)
性能零额外开销每次方法调用需 vtable 查表 + 跳转
能否放不同类型一起?不能可以放进 Vec<Box<dyn Trait>>

7️⃣ Marker Trait——没有函数体的“隐形证书”

证书表示能力典型场景
Copy按位复制标量、小 struct
Send可以安全转到别的线程thread::spawn 移动所有权
Sync多线程可安全共享 &T只读全局配置
Unpin指针可被移动(异步 Pin 相关)自写 Future
// 手动给自定义队列证明线程安全
struct MyQueue<T>(std::sync::Mutex<Vec<T>>);
unsafe impl<T: Send> Send for MyQueue<T> {}
unsafe impl<T: Send> Sync for MyQueue<T> {}

8️⃣ 孤岛法则 (Orphan Rule) ——“证书只能本岛签发”

先弄明白关键名词
crate:Rust 的 “岛” —— 一个编译单元 / 包。

  • 你当前写代码的包 = 当前 crate
  • std = 标准库 crate
  • serde, tokio 等 = 外部 crate

8.1 四类组合(官方规则,按“岛”归属划分)

组合Type 属于Trait 属于impl 吗?口诀
① 当前 crate + 当前 crate本岛本岛自己人给自己发证,随便
外部 crate + 当前 crate外岛本岛自家证书给外来人发
当前 crate + 外部 crate本岛外岛外岛证书发给自家人
④ 外部 crate + 外部 crate外岛外岛“双外来”禁止——怕撞车

你的说法 “当前 crate / Std / 外部 crate” 可以映射到表中:

  • 标准库外部 crate(你改不了源)。
  • 只要落到 组合④(Type + Trait 都不归你),就违规。

8.2 违反怎么办?——Newtype Pattern

// 想给外部库 FooType 实现外部库 BarTrait,不允许
struct MyFoo(FooType);          // 包一层,本岛 Type
impl BarTrait for MyFoo {}   // 现在是组合③,合法

9️⃣ 彻底掌握 Trait 的三步练法

  1. 抄 & 跑
    • 复制本文示例,边改边看编译器错误,尤其试着删掉 Trait Bound。
  2. 写小工具
    • 写个 Printable Trait,自定义三种类型实现;用 impl Trait 返回打印器。
  3. 读官方文档 &源码
    • IteratorRead 这些经典 Trait 的代码,再画出“证书 → 司机”关系图。

🔚 复盘一句话

Trait = 行为证书;Trait Bound = 入场门票;孤岛法则 = 证书只能在自己岛签发,双外来禁止
把这三件事连起来,就能在写泛型、并发、异步时游刃有余。祝练武顺利!

相关文章:

  • RabbitMQ,添加用户时,出现Erlang cookie不一致,导致添加用户失败的问题解决
  • 防抖与节流的理解与应用
  • 代码随想录算法训练营第三十五天|416. 分割等和子集、698.划分为k个相等的子集、473.火柴拼正方形
  • Docker 安装配置教程(配置国内源)
  • 中间件--ClickHouse-14--案例-3-其他案例思路概述
  • JavaScript 一维数组转二维数组
  • Python实现对大批量Word文档进行批量自动化排版(15)
  • 快速下载Node.js
  • 面向对象程序设计(双语)|| 实验八:字符流(Java版)
  • Vue.js 简介
  • Cribl 对Windows-xml log 进行 -flatten-03
  • 【智驾中的大模型 -3】VLA 在自动驾驶中的应用
  • MFC文件-写MP4
  • 时序预测 | Transformer-LSTM-SVM时间序列预测(Matlab完整源码和数据,适合基础小白研究)
  • 卷积神经网络综述
  • QT实现串口透传的功能
  • 提示词工程快速上手
  • 第一章,HCIA复习
  • OpenCV 图像调整指南
  • 【Python爬虫详解】第二篇:HTML结构的基本分析
  • 内部敏感文件遭万人共享,特朗普政府又曝安全漏洞
  • 深一度|上海半马,展示“体育+”无限可能的路跑狂欢
  • 儿童阅读空间、残疾人友好书店……上海黄浦如何打造城市书房
  • 美政府公布1968年罗伯特·肯尼迪遇刺事件档案
  • 历史学家许福谦逝世,长期致力于魏晋南北朝史研究
  • 不断深化“数字上海”建设!上海市数据发展管理工作领导小组会议举行