4.1.1 基本概念

函数观点

操作符可视为一个函数,用参数,副作用,返回值来分析它

cout<<i<<j<<endl>>操作符

  • 参数:ostream对象和值(value)
  • 副作用:将值输入到ostream中,这里是输入到标准输出(屏幕)中
  • 返回值:返回左边的ostream对象,继续与右边的值结合,这是cout可以链式使用的原因

左值和右值

操作符的返回值可以分为左值(lvalue)和右值(rvalue)

  • 左值:用的是对象的身份(内存中的位置)
  • 右值:用的是对象的值(内容)
  • 基本原则:需要右值时可以用左值代替,反之不可以。

运算符重载

对指定的类重新定义某些操作符的操作,但无法改变其运算对象的个数,优先级和结合律

4.1.2 优先级、结合律与求值顺序

优先级和结合律

  • 优先级:每个操作符都有优先级,高级的先运算
  • 结合律:同级运算符遵循结合律,从左到右运算
  • 可用小括号强制改变运算顺序

求值顺序

书中表述

《C++ Primer(英)》137,对应中文版123:

优先级规定了运算对象的组合方式,但是没有说明运算对象按照什么顺序求值。在大多数情况下,不会明确指定求值的顺序。对于如下的表达式

int i = f1()* f2();

我们知道f1和f2一定会在执行乘法之前被调用,因为毕竟相乘的是这两个函数的返回值。但是我们无法知道到底f1在f2之前调用还是f2在f1之前调用。

因此,当在一个表达式的某处改变了一个变量的值,而在该表达式的另一处又使用到它,可能会产生一个未定义的结果,这是一个隐藏错误。

实践表明(猜想)

但是,在g++ 8.1版本下,编译器可以处理上述情况,输出0 1

个人推测:编译器在执行cout<<i<<" "<<++i<<endl;时,会先从左到右执行指令,直到把所有子表达式为字面量。再从左到右输出,所以可以认为编译器执行了两步:(i=0为例)

  1. 子表达式i++i从左到右转为字面量:cout<<0<<" "<<1<<endl;
  2. 输出0 1

实践验证

#include<iostream>
using namespace std;
int m = 0; //全局变量,f,g,j都可以改变m的值 //三个函数都改变了m的值
int f(); //m++
int g(); //m+=2
int j(); //m*=2
int main(){
//如果f,g,j求值顺序不定的话,该式的输出结果会不同,结果未定义。
//但实际上程序输出19
cout<<f()+g()*j()<<endl;
return 0;
} int f(){
m++;
return m;
}
int g(){
m+=2;
return m;
}
int j(){
m*=2;
return m;
}

按照上述猜想可以解释19这一结果

  • 将子表达式fgj从左到右执行,子表达式转为字面量

    • 执行f后,m = 1
    • 执行g后,m = 3
    • 执行j后,m = 6
    • 表达式变为cout<<1 + 2*3<<endl;
  • 输出19

可能的解释:编译器的优化行为

《C++ Primer(英)》138对应中文版124

所以或许可以认为,在遇到某一表达式的子表达式操作了同一对象且改变了对象的值时,编译器会遵循结合律,对每个字表达式从左到右求值,这一动作是编译器的优化动作,来提高执行效率。

但是否每种编译器都执行此类优化呢?不知道。

所以,如果改变了某个运算对象的值,在表达式的其他地方不要再使用这个运算对象。

一些运算符指定了求值顺序

一般运算符不指定求值顺序,但编译器会对其进行优化。而有些运算符明确指定了求值顺序,前三者都是从左到右。

  • &&:左边的子表达式为真时,才会继续求右边的子表达式的值
  • ||:左边的子表达式为假时,才会继续求右边的子表达式的值
  • ,(逗号)
  • ?:(三目条件判断运算符)

再述求值顺序

如果改变了某个运算对象的值,在表达式的其他地方不要再使用这个运算对象,来避免编译器未定义行为

//编写程序把字符串的所有字符大写
#include<iostream>
#include<string>
using namespace std;
int main(){
string s = "hello, world";
auto it = s.begin();
while(it != s.end()){
*it = toupper(*it++); //右值改变了it,左值又用到了it,该行为结果是未定义的
}
cout<<s<<endl;
return 0;
}

*it = toupper(*it++);可能被解释为

  • *it = toupper(*it);
  • *(it+1) = toupper(*it);

修改后:

#include<iostream>
#include<string>
using namespace std;
int main(){
string s = "hello, world";
auto it = s.begin();
while(it != s.end()){
*it = toupper(*it); //把两个语句分开,避免被修改的对象被重用
it++;
}
cout<<s<<endl;
return 0;
}

4-1 C++运算符基本概念的更多相关文章

  1. c/c++ 重载运算符 基本概念

    重载运算符 基本概念 问题:对于int,float可以进行算数运算,但是对于一个自定义的类的对象进行算术运算,就不知道具体怎么运算了. 所以有了自定义运算符的概念. 1,自定义运算符其实就是一个以op ...

  2. C# 中的 null 包容运算符 “!” —— 概念、由来、用法和注意事项

    在 2020 年的最后一天,博客园发起了一个开源项目:基于 .NET 的博客引擎 fluss,我抽空把源码下载下来看了下,发现在属性的定义中,有很多地方都用到了 null!,如下图所示: 这是什么用法 ...

  3. python类可以截获Python运算符

    类可以截获Python运算符 现在,让我们来看类和模块的第三个主要差别: 运算符重载.简而言之,运算符重载就是让用类写成的对象,可截获并响应用在内置类型上的运算:加法.切片.打印和点号运算等.这只是自 ...

  4. C++学习(12)—— 运算符重载

    运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型 1.加号运算符重载 作用:实现两个自定义数据类型相加的运算 #include <iostream> #i ...

  5. ...扩展运算符+rest参数+call/apply/bind

    之前在set,map里面有提过扩展运算符的概念,但是今天偶然遇到一个问题,类似于扩展运算符的经典用法,突然发现对其了解不是很深,所以再来整理一下扩展运算符的相关知识. 重点:扩展运算符内部调用的是数据 ...

  6. C 语言运算符与分支循环小记

    1. 前导内容 · 使用sizeof()统计各种数据类型占用字节数 · 进制转换问题 · if语句基本语法 · switch语句基本语法 2. 运算符 · 基本概念 · 左值 右值 · 优先级结合性 ...

  7. JS006. 详解自执行函数原理与数据类型的快速转换 (声明语句、表达式、运算符剖析)

    今天的主角: Operator Description 一元正值符 " + "(MDN) 一元运算符, 如果操作数在之前不是number,试图将其转换为number. 圆括号运算符 ...

  8. 一篇文章带你了解Java中的运算符

    前言 在前一篇文章中,壹哥给大家讲解了Java数据类型之间的转换,包括自动类型转换.强制类型转换.隐含的强制类型转换等问题.且在上一篇文章中,我还简单地给大家提到了Java的类型提升.在类型提升的案例 ...

  9. JavaSE自学笔记

    ch03 [Thu Aug 18 2016 11:22:26 GMT+0800] 对象变量与对象之间是指代关系,对象变量并不能完全说明有无对象可用.这种指代关系是通过赋值运算建立起来的.对象变量保存的 ...

  10. Python2.6-原理之类和oop(上)

    来自<python学习手册第四版>第六部分 一.oop:宏伟蓝图(26章) 在这之前的部分中,经常会使用"对象"这个词,其实,到目前为止都是以对象为基础的,在脚本中传递 ...

随机推荐

  1. vscode 下配置 clang

    需要在workspace的文件夹下添加文件: .clang-format 更多参数说明: https://clang.llvm.org/docs/ClangFormatStyleOptions.htm ...

  2. 去除WinForm程序中的Devexpress弹窗

    去除WinForm程序中的Devexpress弹窗 /// <summary> /// 应用程序的主入口点. /// </summary> [STAThread] static ...

  3. Ubuntu 切换显示管理器

    比较流行的显示管理器有: gdm3 - GNOME Display Manager lightdm - Light Display Manager sddm - Simple Desktop Disp ...

  4. Windows10使用MSYS2和VS2019编译FFmpeg详解

    1 环境准备 1.1 安装Visual Studio 2019 这个步骤相对比较简单,不再详细说明. 1.2 安装msys2 首先需要安装msys2环境以及相关的编译依赖项, 官方网址为: https ...

  5. C#必备技能—项目打包

    C#项目打包 开发好一个软件后,交给客户去使用,这时需要对项目进行打包成一个.exe文件,怎么去做?(共三步) 前提 安装扩展:状态栏[扩展]-[管理扩展],搜索setup,点击安装(安装比较慢,等待 ...

  6. SciPy从入门到放弃

    目录 SciPy简介 拟合与优化模块 求最小值 曲线拟合 线性代数模块 统计模块 直方图和概率密度函数 统计检验 SciPy简介 SciPy是一种以NumPy为基础,用于数学.工程及许多其他的科学任务 ...

  7. Coursera Self-driving1, introduction

    有哪些 Sensors? 摄像头和激光雷达,毫米波雷达等 sensor 分类: exteroceptive (surrounding), 有 camera(Resolution, FOV, Dynam ...

  8. Docker 知识梳理及其安装使用

    Docker 介绍 Docker 是一个强大的工具,用于高效开发.打包和部署应用程序.Docker 是一种容器管理服务.Docker 于 2013 年发布.它是开源的,可用于 Windows.macO ...

  9. python脚本之requests库上传文件

    一.前言 在学习的时候,发现有一个AWD的文件上传执行漏洞,突然想着批量对不同靶机进行操作并get_flag.思路简单,但是没构造 过文件上传的requests 的post请求的payload.便记录 ...

  10. 小tips:前端容易读错的单词列表

    排名第一的是width,音标/wɪdθ/,发/i/的音,不是发/ai/的音: hidden音标/ˈhɪdn/发/i/的音,不是发/ai/的音: hide音标/haɪd/,发/ai/的音: float音 ...