Rust 中的继承与代码复用

在学习Rust过程中突然想到怎么实现继承,特别是用于代码复用的继承,于是在网上查了查,发现不是那么简单的。

C++的继承

首先看看c++中是如何做的。

例如要做一个场景结点的Node类和一个Sprite类继承它。

定义一个node基类

struct Node {float x;float y;void move_to(float x, float y) {this->x = x;this->y = y;}virtual void draw() const {printf("node: x = %f, y = %f\n", x, y);}
};

再定义一个子类Sprite,重载draw方法:

struct Sprite: public Node {virtual void draw() const {printf("sprite: x = %f, y = %f\n", x, y);}
};

可以把sprite作为一个Node来使用,并且可以重用Node中的move_to函数:

Node* sprite = new Sprite();
sprite->move_to(10, 10);
sprite->draw();

Rust中的继承

现在要用Rust做同样的事。定义一个Node基类:

struct Node {x: f32,y: f32,
}impl Node {fn draw(&self) {println!("node: x={}, y={}", self.x, self.y)}fn move_to(&mut self, x: f32, y: f32) {self.x = x;self.y = y;}
}

定义子类的时候我们遇到了麻烦:Rust里struct是不能继承的!

struct Sprite: Node;

这么写会报错:

error: `virtual` structs have been removed from the language

virtual struct是什么东西?原来Rust曾经有一个virtual struct的特性可以使struct继承另一个struct,但是被删掉了:(
RFC在这里。现在Rust的struct是不能继承的了。

使用 trait

Rust 里的 trait 是类似于 java 里 interface,可以继承的。我们把 Node 定义为 trait。

trait Node {fn move_to(&mut self, x: f32, y: f32);fn draw(&self);
}

但我们发现没有办法在 Node 中实现 move_to 方法,因为 trait 中不能有成员数据:x, y。

那只好在每个子类中写各自的方法实现,例如我们需要一个空Node类和一个Sprite类:

struct EmptyNode {x: f32,y: f32,
}impl Node for EmptyNode {fn draw(&self) {println!("node: x={}, y={}", self.x, self.y)}fn move_to(&mut self, x: f32, y: f32) {self.x = x;self.y = y;}
}struct Sprite {x: f32,y: f32,
}impl Node for Sprite {fn draw(&self) {println!("sprite: x={}, y={}", self.x, self.y)}fn move_to(&mut self, x: f32, y: f32) {self.x = x;self.y = y;}
}

是不是觉得有大量代码重复了?Sprite只需要重写 draw方法,但要把所有方法都实现一遍。如果要实现很多种 Node,每种都要实现一遍,那就要写吐血了。

组合

组合是一个代码重用的好方法。要重用代码时,组合而且比继承更能体现“has-a”的关系。我们把 Node 重新定义为之前的 struct 基类,然后把 Node 放在 Sprite 中:

struct Node {x: f32,y: f32,
}impl Node {fn draw(&self) {println!("node: x={}, y={}", self.x, self.y)}fn move_to(&mut self, x: f32, y: f32) {self.x = x;self.y = y;}
}struct Sprite {node: Node
}impl Sprite {fn draw(&self) {println!("sprite: x={}, y={}", self.node.x, self.node.y)}fn move_to(&mut self, x: f32, y: f32) {self.node.move_to(x, y);}
}

清爽了不少,美中不足的是还不能省略 move_to 方法,还要手动写一遍,简单调用 Node 中的同名方法。

组合和继承还有一些不同的,比如不能把 Sprite 转型为 Node。

Deref & DerefMut trait

std::ops::Deref 用于重载取值运算符: *。这个重载可以返回其他类型,正好可以解决组合中不能转换类型的问题。

在这个例子中,由于 move_to 的 self 可变的,所以要实现 Deref 和 DerefMut

struct Sprite {node: Node
}impl Sprite {fn draw(&self) {println!("sprite: x={}, y={}", self.node.x, self.node.y)}
}impl Deref for Sprite {type Target = Node;fn deref<'a>(&'a self) -> &'a Node {&self.node}
}impl DerefMut for Sprite {fn deref_mut<'a>(&'a mut self) -> &'a mut Node {&mut self.node}
}

之后就可以把 &Sprite 转换为 &Node

let mut sprite = Sprite{ node: Node { x: 10.0, y: 20.0 } };
let mut sprite_node: &mut Node = &mut sprite;
sprite_node.move_to(100.0, 100.0);

要注意的是对sprite_node的方法调用重载是不起作用的。如果 sprite_node.draw(),调用的还是Node.draw(),而不是Sprite.draw()。

如果要调用子类的方法,必须有子类类型的变量来调用。

let mut sprite = Sprite{ node: Node { x: 10.0, y: 20.0 } };// 这个大括号限制 mut borrow 范围
{let mut sprite_node: &mut Node = &mut sprite;sprite_node.move_to(100.0, 100.0);sprite.node.draw(); // 输出 node: x=100, y=100
} sprite.draw(); // 输出 sprite: x=100, y=100

posted on 2015-11-25 21:19 Redclock 阅读(...) 评论(...) 编辑 收藏

转载于:https://www.cnblogs.com/redclock/p/4995954.html

Rust 中的继承与代码复用相关推荐

  1. [基础]Javascript中的继承示例代码

    面向对象的语言必须具备四个基本特征: 1.封装能力(即允许将基本数据类型的变量或函数放到一个类里,形成类的成员或方法) 2.聚合能力(即允许类里面再包含类,这样可以应付足够复杂的设计) 3.支持继承( ...

  2. 基于上下文的智能化代码复用推荐

    点击上方蓝字关注我们 基于上下文的智能化代码复用推荐 彭鑫1,2, 陈驰1,2, 林云3 1 复旦大学计算机科学技术学院,上海 200438 2 上海市数据科学重点实验室,上海 200438 3 新加 ...

  3. 模板代码复用的三种方式: 宏, 继承, 包含

    模板代码复用 在模板中,可能会遇到以下情况: 多个模板具有完全相同的顶部和底部内容 多个模板中具有相同的模板代码内容,但是内容中部分值不一样 多个模板中具有完全相同的 html 代码块内容 宏 对宏( ...

  4. python中基例_Python python从入门到实践(5) --代码复用

    函数.类与代码复用 代码复用的思想 把代码当成资源进行抽象. 代码资源化:程序代码是一种用来表达计算的"资源" 代码抽象化:使用函数等方法对代码赋予更高级别的定义 代码复用:同一份 ...

  5. java中反复使用代码_Java代码复用规则

    Java代码复用规则 要尽量避免在代码中出现判断语句,来测试一个对象是否某个特定类的实例.通常,如果你需要这么做,那么,重新设计可能会有所帮助.我在工作中遇到这样的一个问题:我们在使用JAVA做XML ...

  6. 2021-11-16派森编程学习笔记“上帝视角下的游戏操盘手” 通过类的继承学会了复用代码,减少代码冗余,提高编程效率。

    派森编程学习笔记"上帝视角下的游戏操盘手" [自学笔记] 继承 在Python中继承是指:在类的基础上,它可以实现现有类的所有功能,并在无需重新编写原来的类的情况下对这些功能进行扩 ...

  7. 《On Java 8》- 面向对象之代码复用(组合、继承、委托)

    文章目录 小结 原文 组合语法 继承语法 委托 结合组合与继承 组合与继承的选择 小结 根据<On Java 8>: 第八章复用 总结 复用指的是代码复用,一般情况下有两种方式:组合.继承 ...

  8. 在Eclipse中的Android项目里实现代码复用

    引言 开发时通常都会有多个项目间代码复用的需求,这时通常的做法是设置项目依赖,让一个项目引用另一个项目,比如在Visual Studio中这样就很容易实现. 而在Eclipse中的Android项目里 ...

  9. python中的代码复用与函数递归

    代码复用与模块化设计 所谓的代复用,顾名思义,也就说同一份代码在需要的时候可以被重复使用,有两种形式:一个是函数,一个是对象. 模块化设计,简单地来讲,就是通过函数或对象封装将程序划分为模块及模块间的 ...

最新文章

  1. SAP MM 工序委外流程初探
  2. 安装win7根证书_最详细图解Windows7x64更新安装教程
  3. mac下对NTFS格式的磁盘进行读写操作
  4. php 正则表达式 x,php – 匹配x正则表达式或y正则表达式
  5. 程序员未来的职业生涯路该怎么走,如何避免35岁中年危机?
  6. 12.11团队任务汇总
  7. plsql只提交存储过程里的事务_plsql 存储过程 事务
  8. 请问:如何在C#简单分布式程序的数据层中为其它层留出很好的接口?????...
  9. C++引用计数(reference counting)技术简介(3)
  10. python交互式shell之jupyter notebook初步安装使用
  11. 【IDEA/SVN】IDEA 从SVN导入项目到本地
  12. 电脑鼠标右键应用卡死、转圈圈、资源管理器关闭等问题
  13. 安卓开发:记事本App
  14. 浅谈金融数据中心的消防安全
  15. bailian.openjudge 2692:假币问题
  16. 机器人与视觉,基于坐标系的运动偏移
  17. 基于vue,安装vux-ui步骤
  18. 打开5555调试端口
  19. properties语法
  20. 微信的野心到底有多可怕

热门文章

  1. spring源码分析第一天------源码分析知识储备
  2. pve安装黑群晖直通硬盘_PVE+lede+DSM网卡硬盘直通+win10
  3. java 异步调用webapi_Async Await异步调用WebApi
  4. java 匿名类型_Java之匿名类讲解
  5. 《零基础》MySQL DELETE 语句(十五)
  6. 35岁学嵌入式合适吗_什么是嵌入式技术?
  7. java获取网络带宽_Linux Java 获取CPU使用率,内存使用率,磁盘IO,网络带宽使用率等等...
  8. C++ 获取类型信息
  9. MATLAB基础教程(11)——二维中的数据可视化(续)
  10. 我的Go+语言初体验——(1)超详细安装教程