C/C++基础----重载运算与类型转换
非成员版本
data1 + data2;
operator+(data1, data2);成员版本
data1 += data2;
data1.operator+=(data2);不建议的重载
逻辑与、逻辑或、逗号的运算对象求值顺序规则无法保留。
&&和||的重载版本也没法保留内置运算符的短路求值属性,两个运算对象总是会被求值。
逗号和取址,已经在C++中定义了其用于类对象是的特殊含义,已经有了内置的含义,一般不应该重载。
有些运算符必须作为成员,有些则作为普通函数更好
- 赋值= 下标[] 调用() 成员访问箭头->必须是成员
- 复合赋值一般应该是成员,但并非必须
- 改变对象状态的运算符或者与给定类型密切相关的运算符,如++,--,解引用通常是成员
- 具有对称性的,可能转换任意一端的运算符对象,如算术、相等性、关系和位运算,通常是非成员。
输出运算符<<
ostream &operator<<(ostream &os, const Sales_data &item)
输出运算符不太考虑格式化操作,使用户有权控制输出细节
与iostream标准库兼容的输入输出运算符必须是非成员函数
- 输入运算符>>
istream &operator>>(istream &is, Sales_data &item)
输入运算符必须处理可能失败的情况(数据类型错误,到底文件尾或遇到输入流其他错误)
- 算术和关系
通常定义为非成员,通常不需要改变运算对象(常量引用),计算得到一个新值。
如定义了复合赋值,最有效的是使用复合赋值来定义算术运算。
Sales_data operator+(const Sales_data &lhs, const Sales_data &rhs)
bool operator==(const Sales_data &lhs, cosnt Sales_data &rhs)
bool operator!=(const Sales_data &lhs, cosnt Sales_data &rhs)
- 关系运算符
1 定义顺序关系,与关联容器对关键字的要求一致(唯一性、传递性、等价性)
2 如果同时含有==的话,则定义一种关系令其与==保持一致
当存在一种唯一可靠的<定义,且和==产生的结果一致时,才定义<
- 赋值运算符
StrVec &StrVec::operator=(initializer_list<string> il)
Sales_data &Sales_data::operater+=(const Sales_data &rhs)
赋值运算符必须是成员函数,复合赋值也通常定义为成员函数。
一般算术运算调用复合赋值,可读写较好
- 下标运算
通常返回引用,最好同时定义常量和非常量版本,必须成员函数
std::string &operator[](std::size_t n) {return elements[n];}
const std::string &operator[](std::size_t n) const {return elements[n];}
- 递增递减
StrBlobPtr& StrBlobPtr::operator++() 前置版本
StrBlobPtr StrBlobPtr::operator++(int) 后置版本
后置可以调用前置来完成,前置版本需检查递增操作的有效性。一般设定为成员函数。
- 成员访问
class StrBlob{
public:
std::string &operator*() const
{ auto p = check(curr, “dereference past end”;
return (*p)[curr];
}
std::string *operator->() const
{
return & this->operator*();
}
}
箭头必须成员,解引用也通常成员
箭头运算符永远不能丢掉成员访问的基本含义
point->mem
point必须是指向类对象的指针或者是重载了operator->的类对象
1是指针,等价于(*point).mem
2是对象,调用point.operator->()的结果来获取mem。如果返回的是指针则执行第1步;如果返回的结果本身重载了->,则重复调用。或者返回错误。
- 调用运算符
函数对象,同时也能储存状态,比普通函数更灵活
同一个对象里可以重载好几个不同版本的调用函数,同时可以改变数据成员来定制不同操作。隐含的this参数呢???看调用的形式
- lambda是未命名类的未命名对象
默认情况下,是一个const成员函数,不能改变它捕获的变量。显式声明为mutable则不是。
产生的类不含默认构造函数、赋值运算符及默认析构函数??
是否含有默认的拷贝/移动构造要视捕获的数据类型而定。
- 标准库定义的函数对象
| 算术 | 关系 | 逻辑 |
|---|---|---|
| plus | equal_to | logical_and |
| minus | not_equal_to | logical_or |
| multiplies | greater | logical_not |
| divides | greater_equal | |
| modulus | less | |
| negate | less_equal |
常用来替换算法中的默认运算符,这些函数对象对指针同样适用。
sort(a.beg,a.end, less<string*>() ); //正确
而用<,则将产生未定义的行为
关联容器使用less<key_type>对元素排序,可以定义一个指针作为关键字的set或map而无须直接声明less
- 可调用对象
函数、函数指针、lambda表达式、bind创建的对象、重载了函数调用运算符的类。
fun &fun和funP打印的地址是一样的。funP可以被赋值,而fun不可以。有两种解释
1函数名与FunP函数指针都是函数指针。fun是一个函数指针常量,funP是一个函数数指针变量。
2函数名和数组名实际上都不是指针,但是,在使用时可以退化成指针,即编译器可以帮助我们实现自动的转换。
既然都是都有指针的效果,为什么要定义函数指针?
二义性问题,如有几个版本add函数,不知道哪个?
起到一定的封装效果,可以提供统一接口。C++虚函数表就是通过函数指针实现。
不同类型调用对象可能共享同一种调用形式
map<string, int(*)(int, int)> binops;能存函数指针,存不了函数对象和lambda
function类 <functional>
可以接受同调用类型的可调用对象
function<int (int, int)> f1=add;
map<string,function<int(int,int)>> binops = {
{“+”, add},
{“-”, std::minus<int>()} };
binops[“+”](10, 5); //调用add(10, 5)
类型转换可以面向任何可以作为返回类型的类型,不允许转换成数据或函数类型。,必须定位为成员函数,通常const。
编译器只能进行一个用户定义的类型转换,但是隐式地用户定义类型转换可以置于一个标准(内置)类型转换之前或之后。
explicit operator int() const {return val;}
static_cast<int>(si)+3;
一个例外,当用作条件时,编译器会将显式的类型转换自动应用于它。
if while do for 与或非 ?:
- 避免二义性
两个类提供相同的类型转换
类定义了多个转换规则
当使用用户定义的类型转换时,如果包含标准类型转换,转换的级别决定了最佳匹配选择
C/C++基础----重载运算与类型转换的更多相关文章
- C++ Primer 5th 第14章 重载运算与类型转换
当运算符作用域类类型的对象时,可以通过运算符重载来重新定义该运算符的含义.重载运算符的意义在于我们和用户能够更简洁的书写和更方便的使用代码. 基本概念 重载的运算符是具有特殊名字的函数:函数名由关键词 ...
- c++ 重载运算与类型转换
1. 基础概念 重载的运算符是具有特殊名字的函数:(重载运算符函数,运算符函数.重载运算符) 依次包含返回类型,函数名(operator=),参数列表,函数体. 只有重载的函数调用运算符operato ...
- 【c++ Prime 学习笔记】第14章 重载运算与类型转换
14.1 基本概念 重载的运算符是特殊的函数:名字由关键字operator后接要定义的算符共同组成,也有返回类型.参数列表.函数体. 重载运算符函数的参数量与该算符作用的运算对象数量一样多 除重载调用 ...
- C++ Primer : 第十四章 : 重载运算与类型转换之重载运算符
重载前须知 重载运算符是特殊的函数,它们的名字由operator和其后要重载的运算符号共同组成. 因为重载运算符时函数, 因此它包含返回值.参数列表和函数体. 对于重载运算符是成员函数时, 它的第一个 ...
- 高放的c++学习笔记之重载运算与类型转换
▲基本概念 (1)重载运算符是具有特殊名字的函数,它们的名字又operator和其后要定义的运算符号共同构成.. (2)对于一个运算符号来说它或者是类的成员,或者至少含有一个类类型的参数. (3)我们 ...
- 【转载】DSP基础--定点小数运算
在FPGA实现算法过程中,大多数情况是用占用资源较少,延迟较低的定点数代替浮点数参与运算.那么浮点与定点数之间的区别以及转换方式是怎么的?下边这篇博文详细说明了这一问题.虽然是针对DSP芯片的,但思想 ...
- 「C语言」数据类型及混合运算与类型转换
深入学习C语言时,有必要先了解一下数据类型的概念,以及它们之间的混合运算与类型转换. 本篇文章便是根据<C语言程序设计教程>和在线翻阅资料后整理而出.(练习题将逐步更新) 目录: ...
- C++ Primer 笔记——重载运算
1.对于二元运算符来说,左侧运算对象传递给第一个参数,而右侧运算对象传递给第二个参数.除了重载的函数调用运算符operator()之外,其他重载元素运算符不能含有默认实参. class test { ...
- [C++ Primer] : 第14章: 重载运算符与类型转换
基本概念 重载运算符是具有特殊名字的函数: 它们的名字由关键字operator和其后要定义的运算符号共同组成. 重载运算符函数的参数数量与该运算符作用的运算对象数量一样多. 对于二元运算符来说, 左侧 ...
随机推荐
- JavaBasic_06
二维数组 二维数组定义格式 格式1 数据类型 变量名 = new 数据类型m; m表示这个二维数组有多少个一维数组 n表示每一个一维数组的元素个数 格式2 灵活性 数据类型 a = new 数据类型m ...
- TP thinkphp 权限管理 权限认证 功能
(如有打扰,请忽略)阿里云ECS大羊群,2U4G低至1.4折,限实名新用户,需要的点吧https://promotion.aliyun.com/ntms/act/vm/aliyun-group/tea ...
- Wood Chipping Text Animation
Blender Tutorial: Wood Chipping Text Animationhttps://www.youtube.com/watch?v=YFmN7eTNfNw 文字建模 木板建模, ...
- doubleclick adx note
1, cid . is billing_id from Main.html#PRETARGETING otherwise creative id will not upload to creati ...
- Android命令行工具学习总结
15.setting命令 setting命令可以很方便的更改系统设置中的参数(如修改系统默认输入法) 安卓Settings模块浅析:https://www.jianshu.com/p/ed8508fe ...
- madlib 集成 hasura graphql-engine 试用
madlib 可以让我们直接在sql 中进行机器学习,集成了强大的sql 能力,以及分析能力,后边会尝试 集成graphql engine ,让功能更强大 docker 镜像准备 使用了一个别人的写好 ...
- 使用uflare/smtp2http 将smtp 转转化为http 请求
uflare/smtp2http 是一个很不错的工具,我们使用这个工具,可以快速的将smtp 服务转换为http 服务 用途实际上挺多的 devops 系统 需要使用smtp的系统(测试) 基于smt ...
- linux下PHP手动添加扩展库
1.进入php源程序目录中的ext目录中,这里存放着各个扩展模块的源代码,选择你需要的模块,比如curl模块: cd curl 执行phpize生成编译文件,phpize在PHP安装目录的bin目录下 ...
- redis String结构
1. 设置c的过期时间为100s 2. psetex的单位为毫秒 10000毫秒 3. getrange 获得字符的范围 4. getset 先获得旧的值,然后设置新的值 5. 设置多个值 6. 获得 ...
- SqlServer :利用快捷键快速查看 字段说明查询及表结构 (小技巧)
1.自定义4个常用的存储过程: sp_select :select * from sp_helpremark :查表的列,列的类型,备注(这里只查询有备注的列) sp_columns1 : 查表所有的 ...