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

Cherno C++学习笔记 P41 运算符与重载

这一篇文章中我们讲一下有关于C++的运算符和对运算符的重载问题。C++当中的运算符operator是代替函数执行某些事情的符号,除了我们熟悉的加减乘除等运算符之外,还有一些特殊的,比如new/delete也是操作符,甚至于逗号“,”、“[]”和“()”其实都是operator运算符。

介绍了运算符之后,我们介绍一下什么是重载。重载运算符,指的是允许定义和重写运算符的功能,是一种非常有用的功能。在C++当中,我们对于运算符有完全的控制权,我们在重载运算符的时候有着极高的自由度,当然这也是C++的特点。它给了我们非常高的自由度,所以不同的人写出的代码性能可以天差地别。

但是我们需要注意的是,运算符重载不能随随便便的使用,只有在涉及某些数学量的加减乘除等操作的时候,我们去重载运算符才能够比较有意义。也就是只有非常有意义的时候,我们才能做这件事。如果没有特别的需求,我们最好不要进行运算符的重载。

举一个简单的例子,我们在这里定义一个向量结构体:

struct Vector2 {
	float x, y;
};

我们想要定义一个加法计算,那么第一个我们可以选择写一个函数:

struct Vector2 {
	float x, y;

	Vector2(float xi, float yi)
		:x(xi), y(yi){}
	Vector2 Add(const Vector2& v) {
		return Vector2(x + v.x, y + v.y);
	}
};

同理,我们可以再定义一个向量按entry的乘法:

struct Vector2 {
	float x, y;

	Vector2(float xi, float yi)
		:x(xi), y(yi){}
	Vector2 Add(const Vector2& v) {
		return Vector2(x + v.x, y + v.y);
	}
	Vector2 Multiply(const Vector2& v) {
		return Vector2(x * v.x, y * v.y);
	}
};

那么我们在使用向量加法/乘法的时候,就需要调用函数进行,如果只是一重计算的话还可以,但是如果是多重计算就会比较难绷,比如这个样子:

Vector2 v1(2.2f, 1.1f);
Vector2 v2(0.8f, 3.0f);
Vector2 v3(1.5f, 2.8f);
Vector2 res = v1.Add(v2.Multiply(v3.Add(v1)));

如果写的太长了,最后自己都不知道自己在这里算什么。因此我们需要重载一下向量的运算符号,来让我们的代码变得更加具有可读性。具体重载的方法如下所示:

Vector2 operator+(const Vector2& v) const{
	return Vector2(x + v.x, y + v.y);
}

注意我们需要把这个函数定义在类里面。当然我们也可以直接调用写好了的Add函数:

Vector2 operator+(const Vector2& v) const{
	return Add(v);
}

其实这两个功能是重复的,但是因为使用方式不同,所以通常我们还是会留着两个函数在这里,且直接调用看起来更简洁一些。

同理,我们对乘法也可以做这些操作,那么最后写完的类看起来就如下所示:

struct Vector2 {
	float x, y;

	Vector2(float xi, float yi)
		:x(xi), y(yi){}
	Vector2 Add(const Vector2& v) const{
		return Vector2(x + v.x, y + v.y);
	}
	Vector2 Multiply(const Vector2& v) const{
		return Vector2(x * v.x, y * v.y);
	}
	Vector2 operator+(const Vector2& v) const{
		return Add(v);
	}
	Vector2 operator*(const Vector2& v) const {
		return Multiply(v);
	}
};

那么刚才我们的计算公式就可以写成下面这个更加显然的形式:

Vector2 res = v1 + v2 * (v3 + v1);

这样看起来是不是清晰多了!

另一个比较有意义的重载是重载“==”,也就是判断两个向量是否相等的功能,如下所示(不过同样需要定义在类里面):

bool operator==(const Vector2& v) const {
	return x == v.x && y == v.y;
}

这样我们直接使用==就可以判断两个Vector2类型的变量是否相等了!同理,我们也可以重载“!=”:

bool operator!=(const Vector2& v) const {
	return !(*this == v);
}

最后我们重载一下一个我们经常会使用的操作符,也就是cout当中的“<<”,表示输出流的操作符。但是因为这个操作符所返回的类型其实是std::ostream,而我们需要在标准库的类定义外面去定义这个函数,所以我们需要写成如下形式:

static std::ostream& operator<<(std::ostream &stream, const Vector2& v) {
	stream << v.x << ", " << v.y << std::endl;
	return stream;
}

这样的话我们就可以直接用<<来输出我们的向量了!

这就是有关于操作符重载的全部内容,但是需要注意的是,如果重造操作符会降低我们代码的可读性,那就不要这样做!


http://www.kler.cn/a/441939.html

相关文章:

  • HTML知识点复习
  • Vue3 nginx 打包后遇到的问题
  • 一些面试常见问题及其回答参考
  • 力扣hot100之螺旋矩阵
  • “libcudart,so.1 1.0“ loss解决方案
  • Ubuntu 22.04 TLS 忘记root密码,重启修改的解决办法
  • Elasticsearch:使用 Open Crawler 和 semantic text 进行语义搜索
  • 华为HarmonyOS帮助应用实现在线认证服务 -- 2 FIDO免密身份认证
  • [Unity]在unity 中输出调试安卓真机日志
  • react Ant Design
  • Next.js搜索引擎优化:静态站点生成的SEO优势详解
  • 【C语言】打牌游戏
  • visual studio添加滚动条预览
  • SSM 寝室管理系统:为住宿生活保驾护航
  • sqlilabs第三十关到第三十五关靶场攻略
  • Linux脚本语言学习--上
  • Cesium-(Primitive)-(CylinderGeometry)
  • 8、基于SpringBoot的房屋租赁系统
  • Nginx负载均衡策略详解
  • 安装与配置MongoDB 6.0以支持远程连接
  • 搭建Tomcat(六)---Response的实现
  • 为什么 HTTP/3 抛弃了 TCP?是解决问题还是制造问题
  • Linux——Shell
  • 什么是 PHP 键值对
  • Java 多态的理解
  • 按类别调整目标检测标注框的写入顺序以优化人工审核效率