C++:使用已实现的函数创建“抽象类”

如何解决C++:使用已实现的函数创建“抽象类”

我创建了一个类来表示向量 (Vec3D),以及它的基本规范运算(求和、标量积、叉积等)。我将使用这个类来推导真实物理单位的类(位置、速度、加速度、力等),这些类将具有自己的“专有”特性和操作,*除了它们的矢量操作 .

我希望 Vec3D 是不可实例化的,因为一切都必须是一个物理单位,但使其抽象意味着我必须再次为每个派生类重新定义所有运算符。

这里的最佳实践是什么,它允许我利用已经定义的运算符? (也许将构造函数定义为私有,将运算符定义为公共?

在什么情况下我需要使用虚函数?

在我的基类下面:

class Vec3D
{
private:
  double _x;
  double _y;
  double _z;

protected:
  // Constructors / Destructor
  Vec3D(const double x,const double y,const double z);
  Vec3D();
  Vec3D(const Vec3D& source);
  Vec3D(Vec3D&& source);
  ~Vec3D(); // Declared just for completeness,although not necessary

public:
  // Operator Overloading
  Vec3D& operator=(const Vec3D& source);
  Vec3D& operator=(Vec3D&& source);
  Vec3D& operator+=(const Vec3D& source);
  Vec3D& operator+=(Vec3D&& source);
  Vec3D operator+(const Vec3D& source) const;
  Vec3D operator+(Vec3D&& source) const;
  Vec3D operator-() const;
  Vec3D& operator-=(const Vec3D& source);
  Vec3D& operator-=(Vec3D&& source);
  Vec3D operator-(const Vec3D& source)const;
  Vec3D operator-(Vec3D&& source) const;
  Vec3D operator*(const double source) const;
  double operator*(const Vec3D& source) const;
  double operator*(Vec3D&& source) const;
  Vec3D operator^(const Vec3D& source) const;
  Vec3D operator^(Vec3D&& source) const;
};

谢谢

解决方法

为了解决您的问题,您可以做一些类似于我最近为工作所做的事情。

我使用了继承的组合,使用 Curiously Recurring Template Pattern (CRTP) 作为 mixin,然后是赋值运算符的宏。

对于您的情况,您的赋值运算符可以使用普通的受保护成员函数将实际实现卸载到基类。

也许是这样的:

template<typename RealVec>
class VecBase
{
public:
    friend RealVec operator+(RealVec lhs,RealVec const& rhs)
    {
        return lhs += rhs;
    }

    friend RealVec operator*(RealVec lhs,RealVec const& rhs)
    {
        return lhs *= rhs;
    }

protected:
    // Compound addition and assignment
    RealVec& add_assign_this(RealVec& lhs,RealVec const& rhs)
    {
        // To prevent self-assignment
        if (&lhs != &rhs)
        {
            lhs._x += rhs._x;
            lhs._y += rhs._y;
            lhs._z += rhs._z;
        }
        return lhs;
    }

    // Compound multiplication and assignment
    RealVec& mul_assign_this(RealVec& lhs,RealVec const& rhs)
    {
        // To prevent self-assignment
        if (&lhs != &rhs)
        {
            lhs._x *= rhs._x;
            lhs._y *= rhs._y;
            lhs._z *= rhs._z;
        }
        return lhs;
    }

    // Plain assignment,=
    RealVec& assign_this(RealVec& lhs,RealVec const& rhs)
    {
        // To prevent self-assignment
        if (&lhs != &rhs)
        {
            static_cast<VecBase&>(lhs) = rhs;
        }
        return lhs;
    }

    // Protected constructors to inhibit instantiation
    VecBase()
        : _x{ 0 },_y{ 0 },_z{ 0 }
    {
    }

    VecBase(int x,int y,int z)
        : _x{ x },_y{ y },_z{ z }
    {
    }

    VecBase(VecBase const&) = default;
    VecBase(VecBase&&) = default;

private:
    int _x;
    int _y;
    int _z;

    // The default assignment operator will do the correct thing,no need to implement it
    VecBase& operator=(VecBase const&) = default;
};

class Vector3D : public VecBase<Vector3D>
{
public:
    // Unfortunately we need to reimplement the constructors,// but can "call" the base constructors
    Vector3D()
        : VecBase<Vector3D>{ }
    {
    }

    Vector3D(int x,int z)
        : VecBase<Vector3D>{ x,y,z }
    {
    }

    Vector3D(Vector3D const& other)
        : VecBase<Vector3D>(other)
    {
    }


#define DEFINE_COMPOUND_ASSIGNMENT_OPERATOR(oper,func)           \
    Vector3D& operator oper (Vector3D const& rhs)                 \
    {                                                             \
        return VecBase<Vector3D>::func##_assign_this(*this,rhs); \
    }

    DEFINE_COMPOUND_ASSIGNMENT_OPERATOR(+=,add)
    DEFINE_COMPOUND_ASSIGNMENT_OPERATOR(*=,mul)

#undef DEFINE_COMPOUND_ASSIGNMENT_OPERATOR

    // We still need to explicitly define the plain assignment operator
    Vector3D& operator=(Vector3D const& rhs)
    {
        return VecBase<Vector3D>::assign_this(*this,rhs);
    }
};

[See it running on the Compiler Explorer site]

在继承的类中还有很多要写,遗憾的是真的无法避免。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)&gt; insert overwrite table dwd_trade_cart_add_inc &gt; select data.id, &gt; data.user_id, &gt; data.course_id, &gt; date_format(
错误1 hive (edu)&gt; insert into huanhuan values(1,&#39;haoge&#39;); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive&gt; show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 &lt;configuration&gt; &lt;property&gt; &lt;name&gt;yarn.nodemanager.res