Rust 中的继承与代码复用

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/574163.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

怎么将自己的头像p到特定的背景图_【后期修图】photoshop手把手教你制作属于自己的酷炫的微信头像...

原标题&#xff1a;【后期修图】photoshop手把手教你制作属于自己的酷炫的微信头像效果图先放上1.新建一个白色的背景&#xff0c;这里是1080*720的。然后在图层面板下面单击创建新的调整图层/渐变&#xff0c;设 置 好渐变参数&#xff1a;黑白渐变&#xff0c;径向&#xff0…

10双屏鼠标过不去_灵耀X2 Duo双屏笔记本是怎样“炼”成的?对话华硕笔记本设计团队...

2020年的CES上&#xff0c;华硕推出了一款灵耀X2 Duo笔记本&#xff0c;灵耀X2 Duo采用了和灵耀X2 Pro相同的屏幕技术&#xff0c;C面同样采用了一块ScreenPad Plus副屏&#xff0c;机身也更轻更薄。如今灵耀X2 Duo的处理器升级至英特尔第十代酷睿&#xff0c;正在参与“618”大…

powershell加win的dns服务器,解决网站负载均衡问题

用我发明的powershell填坑法&#xff0c;加windows的dns服务器。从调整dns服务器解析ip时间段的角度&#xff0c;解决网站负载均衡问题。 ------------------------win2012r2中&#xff0c;用powershell管理dns服务器的&#xff0c;命令列表-------------------------------- 共…

csv python 只写一次_在Python CSV Writer循环中写入一次头

下面是一个scraper&#xff0c;它遍历两个网站&#xff0c;获取团队的花名册信息&#xff0c;将信息放入一个数组中&#xff0c;然后将数组导出到CSV文件中。一切都很好&#xff0c;但唯一的问题是每次scraper移动到第二个网站时&#xff0c;csv文件中的writerow头会重复出现。…

理解 OpenStack 高可用(HA) (6): MySQL HA

本系列会分析OpenStack 的高可用性&#xff08;HA&#xff09;概念和解决方案&#xff1a; &#xff08;1&#xff09;OpenStack 高可用方案概述 &#xff08;2&#xff09;Neutron L3 Agent HA - VRRP &#xff08;虚拟路由冗余协议&#xff09; &#xff08;3&#xff09;Neu…

聚类中心坐标公式如何使用_如何使用CAD看图软件来测量坐标?

当我们在拿到图纸的时候&#xff0c;经常会需要使用测量工具&#xff0c;再次测量dwg图纸上的数据。浩辰CAD看图王软件&#xff0c;不仅可以看图&#xff0c;也可以编辑图纸&#xff0c;测量功能更是很方便&#xff0c;当我们在测量的时候&#xff0c;如何测量图纸上点的坐标呢…

linunx 定位最耗资源的进程

[oracletopbox bdump]$ ps -ef|grep “(LOCALNO)”|sort -rn -k 8,8|head -10oracle 9402 1 67 09:18 ? 00:43:09 oracletopbox (LOCALNO)oracle 15524 12159 0 10:21 pts/2 00:00:00 grep (LOCALNO)oracle 15519 1 0 10:20 ? 00:00:00 or…

德利捷读码器设置软件_S7-1500PLC+变频器+编码器组成位置控制系统

PLC学习中组成部分&#xff1a;位置控制、力矩控制、惯量控制。位置控制模式S7-1500 的运动控制功能支持轴的定位和移动&#xff0c;是 S7-1500 系列 CPU 众多集成功能中的重要组成部分。运动控制功能支持旋转轴、定位轴、同步轴和外部编码器等工艺对象。根据PLC-Open&#xff…

解决 c3p0 和 MySQL 集成情况下,连接长时间闲置后重新使用时报错的问题

MySQL 的默认设置下&#xff0c;当一个连接的空闲时间超过8小时后&#xff0c;MySQL 就会断开该连接&#xff0c;而 c3p0 连接池则以为该被断开的连接依然有效。在这种情况下&#xff0c;如果客户端代码向 c3p0 连接池请求连接的话&#xff0c;连接池就会把已经失效的连接返回给…

汇编语言基础知识摘要(《汇编语言》王爽)第 1 / 17 章

CPU&#xff0c;是计算机的核心部件&#xff0c;它控制整个计算机的运作并进行运算。存储器&#xff0c;即内存&#xff0c;存放CPU需要的指令和数据。寄存器&#xff0c;是CPU中可以储存数据的器件&#xff0c;一个CPU中有多个寄存器。汇编语言由以下3类指令组成&#xff1a; …

vue 添加全局组件_自定义vue2.0全局组件(下篇)

在上篇中&#xff0c;老K为大家介绍了一个初级自定义按钮组件的编写方法。虽然能用&#xff0c;但是还不算完美&#xff0c;可扩展性不够强大。在这一篇中&#xff0c;老K继续为大家完善这个按钮组件。启动命令窗口&#xff0c; 进入在上篇中我们搭建的vue目录中&#xff0c;输…

dijkstra算法matlab程序_Dijkstra算法例子

在Dijkstra算法代码下载本文涉及到的代码。程序代码Dijkstra算法的程序如下&#xff1a;function [d, p] dijkstra(adj, s, t) % 使用dijkstra求最短路径 % adj 输入 矩阵 邻接矩阵 % s 输入 整数 起点 % t 输入 整数 或 [] 终…

hdu 3836 Equivalent Sets

题目连接 http://acm.hdu.edu.cn/showproblem.php?pid3836 Equivalent Sets Description To prove two sets A and B are equivalent, we can first prove A is a subset of B, and then prove B is a subset of A, so finally we got that these two sets are equivalent.Y…

mysql-5.7.24 linux下载_Linux下安装mysql-5.7.24

本文档讲解安装版本为mysql-5.7.24&#xff0c;对于5.7.24之后的版本&#xff0c;不适用此说明文档&#xff0c;主要原因在于之后版本的mysql配置文件的目录位置和结构有所改变&#xff0c;使用此说明可能会出现找不到配置文件或者配置后不生效的情况。安装过程中务必保证文件路…

Spring MVC拦截器+注解方式实现防止表单重复提交

原理&#xff1a;在新建页面中Session保存token随机码&#xff0c;当保存时验证&#xff0c;通过后删除&#xff0c;当再次点击保存时由于服务器端的Session中已经不存在了&#xff0c;所有无法验证通过。 注&#xff0c;如果是集群的方式&#xff0c;则需要将token放入到缓存中…

JavaScript学习笔记(一)—— 数据类型

一、基本数据类型&#xff1a;1、js中基本数据类型:String、Boolean、Number、Null、undefined&#xff1b;复合数据类型&#xff1a;object对象类型、Array数组类型特殊对象&#xff1a;function函数类型基本数据类型① String —字符串。若使用var声明变量&#xff0c;var x…

mysql分组失效_请教MySql中使用表子查询时,试着先排序后分组,出现排序失效的原因?...

1,今天试着码了一下教程里的题目,是找出每一个班级的身高最高的学生,用的是先order by降序排序所有学生升高,再用group by分组每一个班级取第一个值,却发现当使用子查询时,得到的仍旧是未排序时的结果,即没有取到最高身高,想问下为何使用子查询时,order by会失效,谢谢!↑这是建…

MySQL后初次密码_mysql第一次安装成功后初始化密码操作步骤

把文件解压到一个目录下这是解压后的目录将my.ini文件考进去双击打开my.ini找到这两行更改成自己的解压路径保存右键此电脑属性找到高级系统设置配置环境变量环境变量 新建 变量值是解压文件的路径Path 单击path编辑新建之后 用管理员身份打开cmd进入文件路径打开命令行窗口&am…

JQuery 操作 radio 被坑一例

.removeAttr(checked); .prop(checked,false); .prop(checked,true); 与 .attr("checked",true); .attr("checked",true); 有本质区别&#xff0c;上边的方法效果理想。转载于:https://www.cnblogs.com/rgqancy/p/5017357.html

mysql技术简介_mysql的简介和使用

mysql简介数据的所有存储&#xff0c;检索&#xff0c;管理和处理实际上是由数据库软件--DBMS(数据库管理系统)完成的mysql是一种DBMS&#xff0c;即它是一种数据库软件mysql工具mysql是一个客户机-服务器的DBMS。因此为了使用mysql&#xff0c;需要一个客户机&#xff0c;即提…