Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >到其他已知的超类

到其他已知的超类
EN

Stack Overflow用户
提问于 2021-03-28 10:07:17
回答 2查看 120关注 0票数 0

我有一个从另外两个(虚拟)类继承的类。让我们调用我的两个超类A和B。我可以有只从A继承的对象,只从B继承的对象,以及从A和B继承的对象。现在,我希望在某个容器中存放已知从A和B继承的对象:为C/Java语法mashup道歉,但这可能是这样的:std::vector<? : A, B>。现在,这是不存在的,但是事实证明,保存我的容器的类主要使用这些类的"A“功能,这就是我目前使用std::vector<A*>的原因。

然而,在我的代码中的另一个地方,在另一个地方,在不同的地方,访问相同的向量,但目的是使用"B“功能。

虽然我个人知道我已经确保了添加到我的向量中的所有实例都是从A和B扩展的,但是编译器没有。这显然使我很难访问B功能。

因此,我的问题是:我如何从A到B“旁白”?我有一个指向A的指针,它的动态类型类似于C (即,继承自A和B-但不一定是特定的C),但我希望获得指向B的指针,显然不需要更改动态类型。

dynamic_cast做了这项工作,但我认为这是一种丑陋的做法。有更好的主意吗?

EN

回答 2

Stack Overflow用户

发布于 2021-03-28 15:00:49

进行侧播的方法是使用dynamic_cast。(你甚至可以对此进行测试。前往cppreference.com搜索“侧播”。只有一次命中。dynamic_cast将使用它的动态魔力来确保指向对象实际上继承了新类(例如,B),并动态地确定需要向指针中添加什么偏移量来执行侧广播。

我不认为dynamic_cast方法“丑陋”,但我确实发现它缺乏优雅。可以接受,但不是很想要。我更倾向于一种更结构化的方法,因为有人断言所讨论的对象都是从AB继承的。(如果不是因为这个断言,dynamic_cast可能是正确的方法。)

在某个容器中,有一个容器保存已知从AB继承的对象。这样的物品有名字吗?如果是这样的话,这可能是一个新类的基础,称为AB,它继承了AB。它可能只会继承这两个类,但它仍然可以在容器中发挥有用的作用。如果容器要保存指向AB的指针,那么容器将自动记录所有元素必须同时从AB派生的文档。此外,不需要动态检查类型,也不需要处理侧广播失败。编译器将强制要求对象必须同时从AB派生,才能首先添加到容器中。此外,不需要动态计算偏移量,因为该偏移量被放入AB中。总体结果是代码更干净,在编译时进行更多的检查。我更喜欢这个解决方案。

OP的一条评论认为,这个提议的AB“基本上是空的,只能充当包装器,这有点难看”。我不同意“丑陋”这一说法,尽管我并不是说这种观点是错误的。不过,我要指出,这是有先例的。就公共接口而言,std::iostream基本上是完全空的,只用作std::istreamstd::ostream的包装器。它的作用类似于我对AB的建议。欢迎您认为std::iostream是丑陋的,但这是无可否认的标准。

票数 0
EN

Stack Overflow用户

发布于 2021-03-28 15:37:40

不能保证AB在对象中的地址是相同的,甚至对于所有的CDEF,这些地址之间的差异也是一样的。这意味着您需要为向量的每个元素提供额外的数据,这将告诉您如何获得指向基类A和B的指针。

dynamic_cast从虚拟表中提取这些附加数据(假设编译器通过虚拟表实现RTTI )。

优点:

  • 没有内存开销。
  • 可伸缩的,如果您想要添加第三个、第四个等等基类。

缺点:

  • 需要RTTI。
  • dynamic_cast可能有些慢(如果它是通过遍历类层次结构实现的)。

可能的替代办法:

1.适配器

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class Adapter {
 public:
  template<class C>
  /* implicit */ Adapter(C* pointer):
    pointer_a_(pointer),
    pointer_b_(pointer) {
        static_assert(std::is_base_of_v<A, C> && std::is_base_of_v<B, C>,
                      "C should be derived from both A and B.");
  }

  A* getA() const { return pointer_a_; }
  B* getB() const { return pointer_b_; }

 private:
  A* pointer_a_ = nullptr;
  B* pointer_b_ = nullptr;
};

int main() {
  std::vector<Adapter> my_vector;
  C1 c1;
  C2 c2;
  my_vector.emplace_back(&c1);
  my_vector.emplace_back(&c2);
  for (Adapter& elem : my_vector) {
    elem->getA()->foo();
    elem->getB()->bar();
  }
}

优点:

  • 不需要RTTI。
  • 快地。
  • 可以确保在编译时向量中的对象实际上是从AB派生的。

缺点:

  • 来自附加指针的内存开销(每个对象)。

2.添加虚拟函数B* A::getB()。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class A {
 public:
  // Return this (casted to B*), or nullptr if the dynamic object is not derived from B.
  virtual B* getB() { return nullptr; }
  // ...
};

class C : public FooWhichInheritsB, public A {
 public:
  B* getB() override { return this; }
};

int main() {
  std::vector<A*> my_vector;
  C1 c1;
  C2 c2;
  my_vector.emplace_back(&c1);
  my_vector.emplace_back(&c2);
  for (A* elem : my_vector) {
    elem->foo();
    elem->getB()->bar();
  }
}

优点:

  • 不需要RTTI。
  • 没有内存开销(每个对象)。
  • 调用虚拟函数getB()比执行dynamic_cast更快。

缺点:

  • 调用虚拟函数getB()比简单地读取Adapter实现中的pointer_b_成员要慢。
  • 您需要记住在每个CDEF中重写D,.

3.引入一个公共基类AB

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class AB : public A, public B {};
class C : public AB { /* ... */ };
class D : public AB { /* ... */ };

优点:

  • 不需要RTTI。
  • 没有内存开销(每个对象)。
  • 甚至比适配器还快。

缺点:

  • 在类层次结构上添加约束--您不能再向向量中添加class Foo : public A2, public B {};类型的对象(其中A2是从A派生的)。
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/66844580

复制
相关文章
如何在Flask中实现可视化?
首先,我们web端想要去显示一些可视化的数据,我们肯定调用别人写好的库是最好的,有哪些呢?
Python进击者
2021/05/31
1.5K0
Flask 模板控制语句 if for
Flask在模板中有常用的几种控制语句: if控制语句 for控制语句 下面来看看示例加强理解,如下: 模板中的if控制语句 1. 示例视图函数 @app.route('/user') def user(): user = 'libai' return render_template('user.html',user=user) 2.示例模板 <html> <head> {% if user %} <title> hello {{user}} </title>
Devops海洋的渔夫
2019/12/24
5670
在CentOS(Linux)中添加单个IP和批量添加多个IP地方法
1、普通方法: 在/etc/sysconfig/network- s/中新建文件ifcfg-eth0-range0
Inkedus
2020/04/16
4.6K0
在CentOS(Linux)中添加单个IP和批量添加多个IP地方法
在/etc/sysconfig/network- s/中新建文件ifcfg-eth0:*,*为数字序号,多个IP则依次增大 以0为例,建立文件ifcfg-eth0:0
雾海梦曦
2022/11/14
3.7K0
如何在系统中添加字体(添加字体到系统)
笔者最近在使用win10自带的OneNote笔记本记笔记的时候,发现笔者电脑中没有华文新魏这个字体,最开始以为是OneNote不带有这个字体,经过一段时间的收集资料后发现,是笔者电脑win10系统中不带有这个字体,现将有关概念和字体安装方法做一下记录。
全栈程序员站长
2022/08/01
3.9K0
如何在系统中添加字体(添加字体到系统)
winfrom如何在listview中添加控件
private Button btn = new Button(); private void Form1_Load(object sender, EventArgs e) { ListViewItem[] lvs = new ListViewItem[3]; lvs[0] = new ListViewItem(new string[] { "行1列1", "行1列2", "" }); l
码农阿宇
2018/04/18
2.6K0
如何在Hue中添加Spark Notebook
CDH集群中可以使用Hue访问Hive、Impala、HBase、Solr等,在Hue3.8版本后也提供了Notebook组件(支持R、Scala及python语言),但在CDH中Hue默认是没有启用Spark的Notebook,使用Notebook运行Spark代码则依赖Livy服务。在前面Fayson也介绍了《Livy,基于Apache Spark的开源REST服务,加入Cloudera Labs》、《如何编译Livy并在非Kerberos环境的CDH集群中安装》、《如何通过Livy的RESTful API接口向非Kerberos环境的CDH集群提交作业》、《如何在Kerberos环境的CDH集群部署Livy》、《如何通过Livy的RESTful API接口向Kerberos环境的CDH集群提交作业》、《如何打包Livy和Zeppelin的Parcel包》和《如何在CM中使用Parcel包部署Livy及验证》,本篇文章Fayson主要介绍如何在Hue中添加Notebook组件并集成Spark。
Fayson
2018/11/16
6.8K1
如何在python文件中测试sql语句
在manage.py的同级目录下新建一个run.py import os if __name__ == '__main__': #加载Django项目的配置信息 os.environ.setdefault("DJANGO_SETTINGS_MODULE","myblog.settings") #导入Django,并启动Django项目 import django django.setup()  #导入相应的models from person import models  #测试s
西西嘛呦
2020/08/26
1.8K0
mysql添加索引语句_mysql 添加索引语句
mysql>ALTER TABLE `table_name` ADD PRIMARY KEY ( `column` )
全栈程序员站长
2022/09/05
5.5K0
如何在ModelSim中添加Xilinx仿真库
今天给大侠带来在FPGA设计应用中如何在ModelSim中添加Xilinx仿真库,话不多说,上货。
FPGA技术江湖
2020/12/30
5.2K1
如何在Vue中动态添加类名
能够向组件添加动态类名是非常强大的功能。它使我们可以更轻松地编写自定义主题,根据组件的状态添加类,还可以编写依赖于样式的组件的不同变体。
前端小智@大迁世界
2020/05/12
6.2K0
如何在 PHP 中运行 bind_param() 语句?
在PHP中,bind_param()函数是一种准备SQL语句并绑定参数的方法。它通常与预处理语句(prepared statements)一起使用,用于执行数据库操作。bind_param()函数可防止SQL注入攻击,并帮助提高代码的安全性。
七辰
2023/10/04
1.5K0
mpvue小程序单个页面添加config配置[设置标题、下拉刷新、颜色等]
前面有说到wepy和mpvue的区别,所以这段时间一直在用mpvue开发了,今天遇到一个问题,mpvue的下拉刷新配置了居然不生效,也是很无赖。
Javanx
2019/09/04
1.3K0
mpvue小程序单个页面添加config配置[设置标题、下拉刷新、颜色等]
20. Flask 模板控制语句 if for
Flask在模板中有常用的几种控制语句: if控制语句 for控制语句 下面来看看示例加强理解,如下: 模板中的if控制语句 1. 示例视图函数 @app.route('/user') def user(): user = 'libai' return render_template('user.html',user=user) 2.示例模板 <html> <head> {% if user %} <title> hello {{user}} </title>
Devops海洋的渔夫
2022/01/14
2310
条码软件中如何在边框上添加文字
很多用户在使用条码标签软件设计制作标签时,会有自己的一些个性化的需要,虽然条码软件不能和作图软件相比,但是很多效果还是可以通过一些小技巧来实现的。比如下面要给大家介绍的把文字压在边框上的效果。如下图所示。
神奇像素科技
2022/03/01
2.7K0
条码软件中如何在边框上添加文字
如何在 TypeScript 中为对象动态添加属性?
在 TypeScript 中,我们经常需要在运行时动态添加属性到对象上。这是因为 TypeScript 是一种静态类型语言,类型系统在编译时会检查代码的类型安全性,所以在编译时我们无法确定对象上将要添加哪些属性。在本文中,我们将讨论如何在 TypeScript 中为对象动态添加属性,以及这样做的一些注意事项。
网络技术联盟站
2023/06/07
11.7K0
如何在README.md文件中添加图片
1、在github上的仓库建立一个存放图片的文件夹,文件夹名字随意。如:img-folder
挥刀北上
2021/03/10
6.1K0
如何在README.md文件中添加图片
如何在Eclipse中添加Tomcat的jar包
右键项目工程,点击Java Build Path 点击Add Library,选择Server Runtime 选择Tomcat版本 此时就看到拥有了Tomcat的jar包了
用户1154259
2018/01/17
2.8K0
如何在Eclipse中添加Tomcat的jar包
如何在WordPress中添加图片放大预览效果
这里分享一个如何在Wordpress中开启点击图片预览的功能。操作也非常的简单,只需要把下面的内容黏贴到对应的文件中即可。实际的效果预览
兔云小新LM
2023/09/11
3710
如何在WordPress中添加图片放大预览效果
点击加载更多

相似问题

如何在flask中添加后台线程?

36

如何在java中模拟单个语句

13

如何在Flask 1.0.4中启用/配置Jinja行语句?

11

如何在单个UPDATE语句(Oracle)中添加正在运行的ID

14

如何在Flask代码中添加链接转发

19
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文