《C++ Primer》第14章 重载运算与类型转换(一)

news2025/1/11 20:07:53

参考资料:

  • 《C++ Primer》第5版
  • 《C++ Primer 习题集》第5版

14.1 基本概念(P490)

重载的运算符是具有特殊名字的函数,其名字有 operator 和要定义的运算符组合而成。和其他函数一样,重载运算符也具有返回类型、参数列表、函数体。

重载运算符函数的参数数量和该运算符的运算对象数量一样多,对于二元运算符来说,左侧运算对象传递给第一个参数,右侧运算对象传递给第二个参数。除了重载调用运算符 operator() 外,其他重载运算符不能有默认实参。

如果一个重载运算符函数是成员函数,则它的第一个运算对象隐式绑定到 this 指针上。

对于一个重载运算符函数来说,它至少含有一个类类型的参数

8ec4dd2082e6596f5f4ef932fa95612

有四个符号( +, -, *, &既是一元运算符也是二元运算符,具体定义哪种运算符由参数数量决定。

重载运算符不改变优先级结合律

直接调用一个重载运算符

我们可以像调用普通函数一样直接调用运算符函数:

data1 + data2;    // 间接调用
operator+(data1, data2);    // 直接调用
data1.operator+=(data2);    // 直接调用(成员函数)

某些运算符不应该被重载

前面提到过,某些运算符规定了求值顺序(如逻辑与、逻辑或、逗号),由于使用重载运算符本质上是一次函数调用,所以这些求值顺序将无法应用到重载运算符上。另外,逻辑与和逻辑或的短路属性在重载运算符中也不能实现。此外,C++ 语言已经定义了逗号取地址运算符作用于类类型时的含义,所以一般情况下我们也不应该重载它们。

总结:通常情况下,不应该重载逗号、取地址、逻辑与、逻辑或运算符。

使用与内置类型一致的含义

如果某些类操作在逻辑上与运算符相关,则它们适合被定义成重载运算符:

  • 如果类执行 IO 操作,则定义移位运算符使其与内置类型的 IO 保持一致。
  • 如果某个类的操作检查相等性,则定义 operator==operator!=
  • 如果类包含一个单序比较操作,则定义 operator< 和其他比较操作。
  • 重载运算符的返回类型应与内置版本的返回类型兼容:逻辑运算符和关系运算符应返回 bool ;算术运算符返回一个类类型的值;赋值和复合赋值运算符返回左侧运算对象的引用。

选择作为成员或非成员

当我们定义重载运算符时,需要确定将其声明为类的成员函数还是普通函数。下面的准则有助于我们做出选择:

  • 赋值、下标、调用、成员访问箭头运算符必须是成员
  • 复合赋值运算符一般是成员
  • 改变对象状态与给定类型密切相关的运算符,如递增、递减、解引用,通常应该是成员
  • 具有对称性的运算符通常应该是非成员函数

关于最后一点,这里着重解释一下。当我们把运算符定义成成员函数时,它的左侧运算对象必须是运算符所属类的一个对象。而如 + 这种具有对称性的运算符,常常会遇到下面这种情况:

string s = "hello";
string t = s + "!";    // 正确
t = "!" + s;

如果 string 的重载 + 是成员函数,那么最后一条语句就是错误的,我们显然不希望这种情况发生。

14.2 输入和输出运算符(P494)

14.2.1 重载输出运算符<<(P494)

通常情况下,输出运算符的第一个形参是一个非常量 ostream 对象的引用,第二个形参是一个常量的引用,返回值为 ostream 形参(引用类型)。

Sales_data的输出运算符

ostream &operator<<(ostream &os, const Sales_data &item){
    os << item.isbn() << " " << item.units_sold << " "
       << item.revenue << " " << item.avg_price();
    return os;
}

输出运算符应尽量减少格式化操作

内置类型的输出运算符不太考虑格式化操作,尤其不会打印换行符

输入输出运算符必须是非成员函数

IO 运算符一般被声明成类的友元。

14.2.2 重载输入运算符>>

Sales_data的输入运算符

istream &operator>>(istream &is, Sales_data &item){
    double price;
    is >> item.bookNo >> item.units_sold >> price;
    if(is)    // 检测输入是否成功
        item.revenue = item.units_sold * price;
    else    // 输入失败,对象被赋予默认状态
        item = Sale_data();
    return is;
}

输入运算符必须处理输入失败的情况

输入时的错误

当读取操作发生错误时,输入运算符应该负责从错误中恢复。

14.3 算术和关系运算符(P497)

通常情况下,我们把算术运算符关系运算符定义成非成员函数,以允许左侧或右侧运算对象进行转换。

算术运算符通常会计算它的两个运算对象并得到一个新值,这个新值常常位于一个局部变量之内,最后返回该局部变量的副本。如果一个类定义了算术运算符,一般也会定义对应的复合赋值运算符,此时最有效的方式是用复合赋值来定义算术运算符

Sales_data
operator+(const Sales_data &lhs, const Sales_data &rhs) {
	Sales_data sum = lhs;
	sum += rhs;
	return sum;
}

14.3.1 相等运算符(P497)

bool operator==(const Sales_data &lhs, const Sales_data &rhs) {
	return lhs.isbn() = rhs.isbn() &&
		lhs.units_sold == rhs.units_sold
		lhs.revenue == rhs.revenue;
}
bool operator!=(const Sales_data &lhs, const Sales_data &rhs) {
	return !(lsh == rhs);
}

14.3.2 关系运算符(P498)

前面提到过,关联容器和一些算法需要用到小于运算符,所以定义 operator< 会比较有用。此时需要注意,如果类同时也含有 == 运算符,应保证:如果两个对象 == ,则不应有一个对象 < 令一个对象成立;如果两个对象 != ,则必有一个对象 < 另外一个对象。

14.4 赋值运算符(P499)

标准库 vector 支持用花括号内的元素列表赋值:

vector<string> v;
v = {"a", "an", "the"};

同样地,我们也为 StrVec 添加这种赋值方法:

class StrVec{
public:
    StrVec &operator=(initializer_list<string>);
};

StrVec &StrVec::operator=(initializer_list<string> il) {
	auto data = alloc_n_copy(il.begin(), il.end());
	free();
	elements = data.first;
	first_free = cap = data.second;
	return *this;
}

复合赋值运算符

复合赋值运算符不必须是类的成员,但一般还是将其设计成成员函数:

Sales_data &Sales_data::operator+=(const Sales_data &rhs) {
	units_sold += rhs.units_sold;
	revenue += rhs.revenue;
	return *this;
}

14.5 下标运算符(P501)

下标运算符必须是成员函数。为了与下标的原始定义兼容,下标运算符通常以所访问元素的引用作为返回值。同时,我们最好同时定义下标运算符的常量版本非常量版本

class StrVec{
public:
    string &operator[](size_t n)
        { return elements[n]; }
    const string &operator[](size_t n) const
        { return elements[n];}
}

14.6 递增和递减运算符(P502)

C++ 并不要求递增和递减运算符必须是类的成员,但因为它们改变所操作对象的状态,所以建议将其设定为成员函数/

定义前置递增/递减运算符

class StrBlobPtr {
public:
	StrBlobPtr &operator++();
	StrBlobPtr &operator--();
};

StrBlobPtr &StrBlobPtr::operator++() {
	check(curr, "increment past end of StrBlobPtr");
	++curr;
	return *this;
}
StrBlobPtr &StrBlobPtr::operator--() {
	// curr为无符号类型,如果curr为0,--后将得到一个很大的正数
	--curr;
	check(curr, "increment past end of StrBlobPtr");
	return *this;
}

区分前置和后置运算符

前置和后置版本使用的是同一个符号,并且运算对象的数量和类型也相同。为了区分前置版本和后置版本,后置版本接受一个额外的、不被使用的 int 类型形参,当我们使用后置运算符时,编译器为这个形参提供值为 0 的实参:

class StrBlobPtr {
public:
	StrBlobPtr &operator++(int);
	StrBlobPtr &operator--(int);
};

// 无需为int形参命名
StrBlobPtr &StrBlobPtr::operator++(int) {
	StrBlobPtr ret = *this;    // 记录当前的值
	++*this;
	return ret;
}
StrBlobPtr &StrBlobPtr::operator--(int) {
	StrBlobPtr ret = *this;    // 记录当前的值
	--*this;
	return ret;
}

显式地调用后置运算符

StrBlobPtr p(a1);
p.operator++(0);    // 显式调用后置版本
p.operator++();    // 显式调用前置版本

14.7 成员访问运算符(P504)

迭代器类和智能指针类常常用到解引用运算符和箭头运算符:

class StrBlobPtr {
public:
	string &operator*() const {
		auto p = check(curr, "dereference past end");
		return (*p)[curr];
	}
	string *operator->() const {
		return &(this->operator*());
	}
    // ...
};
StrBlob a1 = {"hi", "bye", "now"};
StrBlobPtr p(a1);
*p = "okay";

对箭头运算符返回值的限定

对于形如 point->mem 的表达式来说,point 必须是指向类对象的指针或者是一个重载了 operator->类对象,根据 point 类型的不同,point->mem 分别等价于:

(*point).mem;
point.operator()->mem;

point->mem 的执行过程:

  1. 如果 point 是指针,则我们应用内置箭头运算符,表达式等价于 (*point).mem
  2. 如果 point 是定义了 operator-> 的类的一个对象,则我们使用 point.operator->() 的结果来获取 mem 。如果结果是一个指针,则执行第 1 步;否则重复调用当前步骤。

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

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

相关文章

人工智能推动供应链革命的成功

人工智能推动供应链革命的成功 目录 人工智能推动供应链革命的成功一、供应链管理不断变化的面貌二、拥挤的解决方案景观三、踏上人工智能驱动的转型1. 价值创造识别、战略和路线图2. 目标解决方案设计和供应商选择3. 实施与系统集成4. 变革管理、能力建设和全面价值获取 新技术…

深思这届CES,前沿新物种「辣眼睛」背后

作者 | 陈然 来源 | 洞见新研社 每届CES都会让不少人发出“无趣”的感叹。的确&#xff0c;无论是置身会场还是看网络上的报道&#xff0c;到处都是熙熙攘攘的人群&#xff0c;很难从中发现哪些产品或创意值得把玩一番。 实际上&#xff0c;辣眼睛的新物种常有&#xff0c;制…

发动机装备3d虚拟在线云展馆360度展示每处细节

在当今数字化的时代&#xff0c;消费者对于线上购物的需求与期待日益增长。尤其在购车这一大宗消费行为上&#xff0c;消费者不再满足于传统的图片与文字介绍。为了满足这一市场需求&#xff0c;我们引入了3D线上展示技术。 3D汽车模型实景互动展示是一种通过先进的三维建模技术…

软件测试|selenium 元素无此属性NoSuchAttributeException问题分析与解决

简介 在使用Selenium进行Web自动化测试时&#xff0c;我们可能会遇到NoSuchAttributeException异常。这个异常通常在尝试访问一个元素的属性&#xff08;attribute&#xff09;时抛出&#xff0c;但该属性不存在。本文将介绍NoSuchAttributeException异常的常见原因以及解决方…

Exception in thread “main“ java.lang.ArrayIndexOutOfBoundsException(数组创建问题)

数组在Java中使用还是比较多的&#xff0c;通过索引去数组中寻值&#xff0c;也可以通过数组索引去赋值 问题描述&#xff1a; 我们在直接使用未被new的数组时就会出现这种情况&#xff0c; 这边简单创建一个运行类 public class a {public static void main(String[] args)…

大数据-hive函数与mysql函数的辨析及练习-将多行聚合成一行

目录 1. &#x1f959;collect_list: 聚合-不去重 2. &#x1f959;collect_set(col): 聚合-去重 3. &#x1f959;mysql的聚合函数-group_concat 4. leetcode练习题 1. &#x1f959;collect_list: 聚合-不去重 将组内的元素收集成数组 不会去重 2. &#x1f959;collec…

Notepad++安装步骤

Notepad是一款文本编辑工具&#xff0c;支持27种编程语言&#xff0c;通吃C,C ,Java ,C#, XML, HTML, PHP,JS 等&#xff0c;该软件拥有完整的中文化接口及支持多国语言编写的功能&#xff0c;不仅可以用来制作一般的纯文字说明文件&#xff0c;还非常适合编写计算机程序代码&a…

在Windows Server 2012中部署war项目

目录 一.安装jdk 二.安装tomcat 三.安装MySQL 四.部署项目 好啦今天就到这了&#xff0c;希望帮到你了哦 前言&#xff1a;具体步骤&#xff1a; 1.安装JDK&#xff1a; 2.安装tomcat&#xff1a; 3.安装MySQL&#xff1a; 4.部署项目&#xff1a; 一.安装jdk 将所需文件放…

第一波!2024年1月精选6款实用AI人工智能设计工具合集

大家好&#xff0c;这是进入2024年之后的第一波干货合集&#xff01;这次的干货合集还是以 AI 相关的设计干货开头&#xff0c;这次有了在本地无限制帮你清理图片中元素的 AI 工具&#xff0c;有知名免费图库出品的实时 AI 图片生成工具、将截图直接转化为代码的超强工具&#…

x-cmd pkg | smartctl - 用于监测和分析硬盘的工具

目录 简介首次用户功能特点竞品和相关作品进一步阅读 简介 smartctl 是一个用于监测和分析硬盘中 S.M.A.R.T.&#xff08;自我检测&#xff0c;分析和报告技术&#xff09;信息的命令行工具&#xff0c;是 Smartmontools 的一部分。通过 smartctl 工具&#xff0c;可以分析各种…

接口自动化测试介入项目管理流程

上图为接口自动化测试介入梧桐项目管理流程图 前景和目标&#xff1a; 现在公司的项目流程都是全部开发完成后提交到测试环境进行测试&#xff0c;导致测试人员在开发编码过程中相对清闲&#xff0c;除了完成测试用例之外没有其他事情可做&#xff0c;而当进入测试阶段又会变的…

[windows]一种判断exe是32位还是64位程序简单方法

不用运行&#xff0c;直接查看 exe 文件的兼容性属性。 如果是 32 位的程序&#xff0c;“简化的颜色模式”和“用 640x480 屏幕分辨率运行”是可以勾选的&#xff0c;且兼容模式最低可以调到 Windows 95。 而 64 位的程序&#xff0c;“简化的颜色模式”和“用 640 x 480 屏…

什么是reids缓存雪崩、穿透、击穿

1.Reids缓存雪崩 Redis缓存key同一时间大量失效&#xff0c;导致大量请求全部打到数据库&#xff0c;造成数据库挂掉 解决方案 设置缓存失效时间&#xff0c;随机初始化失效时间 部署集群的时候&#xff0c;把热点数据平均分布到不同redis节点上去 暴力方法&#xff0c;不…

Spirng MVC见解1

1. SpringMVC概述 1.1 MVC介绍 MVC是一种设计模式&#xff0c;将软件按照模型、视图、控制器来划分&#xff1a; M&#xff1a;Model&#xff0c;模型层&#xff0c;指工程中的JavaBean&#xff0c;作用是处理数据 JavaBean分为两类&#xff1a; 一类称为数据承载Bean&#x…

x-cmd pkg | trash-cli - 类 Unix 系统的命令行垃圾桶

目录 简介首次用户技术特点竞品和相关作品进一步阅读 简介 trash-cli 是类 Unix 系统的命令行垃圾桶&#xff0c;用于移动文件到回收站&#xff0c;同时会记录文件的原地址和删除日期。 该工具使用与 GNOME、KDE ​​和 XFCE 等桌面环境相同的垃圾桶&#xff0c;所以即使是非 …

数字信号处理实验---Z变换及系统的零极点分析 Matlab代码

一&#xff0e;各种函数的用法 1.tf2zp函数&#xff1a;通常用于将传递函数&#xff08;Transfer Function&#xff09;转换为零极增益形式&#xff08;ZPK form&#xff09;&#xff0c;转换前G(s) num(s) / den(s)&#xff0c;转换后G(s) K * (s - z1) * (s - z2) * ... *…

matlab使用PhysioNet的WFDB工具箱

目录 PhysioNet 官方安装示例 PhysioNet 官方 官方的文档&#xff1a;Waveform Database Software Package (WFDB) for MATLAB and Octave 简介&#xff1a;用于MATLAB的WFDB工具箱是用于读取、写入和操作&#xff08;处理&#xff09;PhysioNet数据的MATLAB函数的集合&#x…

MySQL篇—自带物理克隆数据工具Clone插件介绍(第一篇,总共三篇)

各位小伙伴&#xff0c;今天我为大家介绍一下MySQL Clone Plugin这个插件&#xff0c;简单来说&#xff0c;就是MySQL 8.0.17版本之后的一个物理克隆数据工具&#xff0c;它能够帮助我们快速、高效地克隆或复制数据库&#xff0c;极大地简化了数据库迁移、备份和恢复的过程&…

CSRF漏洞+附pikachu靶场详解

文章目录 前言一、CSRF漏洞是什么二、CSRF漏洞形成的条件1、用户要在登录状态&#xff08;即浏览器保存了该用户的cookie&#xff09;2、用户要访问攻击者发的恶意url链接才行 三、CSRF漏洞复现&#xff08;pikachu靶场&#xff09;1、CSRF&#xff08;get&#xff09;2、CSRF&…

Netty 介绍、使用场景及案例

Netty 介绍、使用场景及案例 1、Netty 介绍 https://github.com/netty/netty Netty是一个高性能、异步事件驱动的网络应用程序框架&#xff0c;用于快速开发可扩展的网络服务器和客户端。它是一个开源项目&#xff0c;最初由JBoss公司开发&#xff0c;现在由社区维护。Netty的…