C++ 系列 第五篇 C++ 算术运算符及类型转换

news2025/1/12 10:46:45

系列文章

C++ 系列 前篇 为什么学习C++ 及学习计划-CSDN博客

C++ 系列 第一篇 开发环境搭建(WSL 方向)-CSDN博客

C++ 系列 第二篇 你真的了解C++吗?本篇带你走进C++的世界-CSDN博客

C++ 系列 第三篇 C++程序的基本结构-CSDN博客

C++ 系列 第四篇 C++ 数据类型上篇—基本类型-CSDN博客 

前言

        我们上一篇总结了C++的基本类型,涉及整形和浮点型,总的来说都是数值类型,在实际编程中一定会碰到数值相关的算术运行。所以这一篇我们总结下C++的算术运算,最主要的是不同类型 的整形或浮点型进行算术运算时候的一些隐式类型转换,虽然是一些基本得知识,但真的是容易被遗忘或者编程时被忽视的点,相信有足够编程经历的程序猿们多少都被类型隐式转换坑过。

C++运算符

        C++使用运算符来运算。提供了几种运算符来完成5种基本的算术计算,分别是 加法(+)、减法(-)、乘法(*)、除法(、)以及求模(%), 每种运算符都使用两个值(操作数)来计算结果,运算符及操作数构成了表达式,变量和常量都可以用作操作数。C++中这些算术运算符的使用方式及限制和C 语言是一致的。

        + 运算符对操作数执行加法运算。例如,4+20 等于 24。

        - 运算符从第一个数中减去第二个数。例如,12-3 等于 9。

        * 运算符将操作数相乘。例如,28 * 4 等于 112。

        / 运算符用第一个数除以第二个数。例如,1000 / 5等于200。如果两个操作数都是整数,则结果为商的整数部分。例如,17 / 3 等于 5,小数部分被丢弃。

        % 运算符求模。也就是说,它生成第一个数除以第二个数后的余数。例如,19 % 6 为 1,因为 19是 6 的 3 倍余 1。两个操作数必须都是整型,将该运算符用于浮点数将导致编译错误。

运算符的优先级

示例1:不同优先级操作符

        int value=3 + 4 * 5;  结果是35还是23呢?

        当多个运算符可用于同一个操作数时,C++使用优先级规则来决定首先使用哪个运算符。算术运算符遵循通常的代数优先级,先乘除,后加减。因此 3+4*5 指的是 3+(4*5),而不是(3+4)*5,结果为 23,而不是 35。当然,可以使用括号来执行自己定义的优先级。

        乘法(*)、除法(/)和 取模(%)优先级相同。同样,加和减的优先级也相同,但比乘除低。

示例2:同优先级操作符

        float value = 120 / 4 *5; 结果是150 还是6呢?

        运算符 "/" 和 "*" 的优先级相同,因此优先级本身并不能指出程序究竟是先计算120除以 4.还是先计算4乘以5。因为第一种选择得到的结果是150,而第二种选择的结果是6,因此选择十分重要。

        当两个运算符的优先级相同时,C++将看操作数的结合性是从左到右,还是从右到左。从左到右的结合性意味着如果两个优先级相同的运算符被同时用于同一个操作数,则首先应用左侧的运算符。从右到左的结合性则首先应用右侧的运算符。C++ 及C 语言中,乘除都是从左到右结合的。这说明应当先对4使用左侧的运算符。也就是说,用120 除以 4,得到的结果为 30,然后再乘以 5,结果为 150。注意,仅当两个运算符被用于同一个操作数时,优先级和结合性规则才有效。

示例3:有优先级和结合性是否就能确保结果?        

        int value = 20 * 5 + 24 * 6

        运算符优先级表明了两点:程序必须在做加法之前计算20 * 5,必须在做加法之前计算24 * 6。但优先级和结合性都没有指出应先计算哪个乘法。读者可能认为,结合性表明应先做左侧的乘法,但是在这种情况下,两个*运算符并没有用于同一个操作数,所以该规则不适用。事实上,C++把这个问题留给了实现,让编译器来决定在系统中的最佳顺序。对于这个例子来说,两种顺序的结果是一样的。

        但是也有两种顺序结果不同的情况,比如后边章节会总结的递增运算符,如下例子中vlaue4 就是一个极端的例子。

类型转换 

        C++及C 语言丰富的类型允许根据需求选择不同的类型,这也使计算机的操作更复杂。例如,将两个 short 值相加涉及到的硬件编译指令可能会与将两个 long 值相加不同。由于有 11 种整型和 3 种浮点类型,因此计算机需要处理大量不同的情况,尤其是对不同的类型进行运算时。为处理这种潜在的混乱,C++及C语言在以下情况中会自动执行类型转换:

        1、将一种算术类型的值赋给另一种算术类型的变量时,将对值进行转换;

        2、表达式中包含不同的类型时,将对值进行转换;

        3、将参数传递给函数时,将对值进行转换。   

        4、使用 和 函数返回类型不同的类型变量接受返回值,将对值进行转换。(这个也是实际编程中经常会碰到的bug)  

初始化及赋值时的类型转换 

        C++允许将一种类型的值赋给另一种类型的变量。这样做时,值将被转换为接受变量的类型,如下示例,long 类型的值赋值给int 型时,转换为了ini 型,double赋值给int 型时,也转换为了int型。

         将一个值赋给值取值范围更大的类型通常不会导致什么问题。例如,将 short 值赋给long 变量并不会改变这个值,只是占用的字节更多而已。然而,将一个很大的long值(如2111222333)赋给float变量将降低精度。因为 float 只有6 位有效数字,因此这个值将被四舍五入为 2.11122E9。因此,有些转换是安全的,有些则会带来麻烦。如下表列出了可能出现的转换问题。

 以{}初始化时进行的转换

        C++11 将使用大括号的初始化称为列表初始化,因为这种初始化常用于给复杂的数据类型提供值列表。它对类型转换的要求更严格。具体地说,列表初始化不允许缩窄(narrowing),即变量的类型可能无法表示赋给它的值。例如,不允许将浮点型转换为整型。将整型转换为浮点型被允许,条件是编译器知道目标变量能够正确地存储赋给它的值。例如,可将 long 变量初始化为 int 值,因为 long 总是至少与 int 一样长;相反方向的转换也可能被允许,只要 int 变量能够存储赋给它的 long 常。

表达式中的转换 

        当同一个表达式中包含两种不同的类型时,C++将执行两种自动转换:首先,一些类型在出现时便会自动转换;其次,有些类型在与其他类型同时出现在表达式中时将被转换。

        先来看看自动转换。在计算表达式时,C++将bool、char、unsigned charsigned char 和 short 值转换为int。这些转换被称为整型提升(integral promotion)。

        我们还是写一个小用例来说明问题,如下我们声明了两个short类型的变量a和b,并打印它们的大小(以字节为单位)。然后,我们将a和b相加,并打印相加后的结果的大小。

        当运行这个程序时,会发现a和b的大小都是 2 字节。然而,当我们将它们相加后,实际相加后结果并不需要int来存储,结果的大小却是 4 字节,这就证明了short类型在表达式计算中被自动提升为int类型。

同样,wchar_t 被提升成为下列类型中第一个宽度足够存储wchar_t取值范围的类型:int、unsigned int、long 或unsigned long.

        再来看下不同类型进行算术运算时,进行的一些转换。

        当运算涉及两种类型时,较小的类型将被转换为较大的类型。例如,用 9.0 除以 5。由于9.0 的类型为 double,因此程序在用5除之前,将 5 转换为 double 类型。运算时的转换,符合以下规律:

        1、如果有一个操作数的类型是 long double,则将另一个操作数转换为 long double。

        2、否则,如果有一个操作数的类型是double,则将另一个操作数转换为double。

        3、否则,如果有一个操作数的类型是 float,则将另一个操作数转换为float。

        4、否则,说明操作数都是整型,因此执行整型提升。

        5、在这种情况下,如果两个操作数都是有符号或无符号的,且其中一个操作数的级别比另一个低,则转换为级别高的类型。这里所说的级别就是 比如long 级别比int 高。

        6、如果一个操作数为有符号的,另一个操作数为无符号的,且无符号操作数的级别比有符号操作数高,则将有符号操作数转换为无符号操作数所属的类型。

        7、否则,如果有符号类型可表示无符号类型的所有可能取值,则将无符号操作数转换为有符号操作数所属的类型。

        8、否则,将两个操作数都转换为有符号类型的无符号版本。

函数涉及的转换

        传递参数时的类型转换通常由C++函数原型控制。函数返回值,则由 接受返回值的变量类型进行确认,在日常代码工程中,经常能看到 有人 封装的一个函数 返回 -1 (int型), 但是 使用的人,没注意看,用的是无符号int 定义的变量 进行接受的,就会导致 整个逻辑判断出错,一定要当心。

强制类型转换

        C++还允许通过强制类型转换机制显式地进行类型转换。强制类型转换的格式有两种。我们还是用实例说明问题,如下示例中, a 是short 型, 可以 使用(typename) variable 或者 typename (variable) 的方式进行强制类型转换,同时强制类型转换不会修改原有变量的类型,而是创建一个新的、指定类型的值,可以在表达式中使用这个值。

        (typename) variable 或者 typename (variable)  第一种格式来自C语言,第二种格式是纯粹的 C++。新格式的想法是,要让强制类型转换就像是函数调用。

        当然C++ 认为C语言的强制转换太危险,所以还增加了其他的约束力更强的强制类型转换,有4中,分别如下,我们这里只做简单 陈述,后边涉及到再展开讨论:

  1. static_cast:用于非多态类型之间的转换,如基本数据类型之间的转换,以及具有继承关系的类型之间的转换。但它不能用于将 const 或 volatile 限定符添加或删除。

  2. dynamic_cast:用于具有继承关系的类型之间的转换,它在运行时进行类型检查,只能用于具有虚函数的类(多态类型)。它可以用于将指向基类的指针或引用转换为指向派生类的指针或引用,或者将指向派生类的指针或引用转换为指向基类的指针或引用。如果转换失败,dynamic_cast 运算符将返回一个空指针(对于指针类型)或引发一个 std::bad_cast 异常(对于引用类型)。

  3. const_cast:用于添加或删除 const 或 volatile 限定符。它可以用于将常量对象转换为非常量对象,或者将非常量对象转换为常量对象。但是,const_cast 并不能用于修改本来就是常量的对象,或者用于修改指向常量对象的指针。

  4. reinterpret_cast:用于不同类型之间的强制类型转换,它可以将任意类型的指针或引用转换为其他类型的指针或引用,甚至可以将指针转换为整数类型,或者将整数类型转换为指针。但是,使用 reinterpret_cast 进行类型转换时需要非常小心,因为它会绕过编译器的类型检查,可能导致未定义的行为。

总结

        C++使用运算符来提供对数字类型的算术运算:加、减、乘、除和求模。当两个运算符对同一个操作数进行操作时,C++的优先级和结合性规则可以确定先执行哪种操作。

        对变量赋值、在运算中使用不同类型、使用强制类型转换时,C++将把值从一种类型转换为另一种类型。很多类型转换都是“安全的”,即可以在不损失和改变数据的情况下完成转换。例如,可以把 int 值转换为 long 值,而不会出现任何问题。对于其他一些转换,如将浮点类型转换为整型,则需要更加小心。

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

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

相关文章

nodejs微信小程序+python+PHP问卷调查系统的设计与实现-计算机毕业设计推荐

目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性:…

从存储器原理看 cpu 内存墙的本质

冯诺伊曼结构是个 cpu 和主存储器的通信结构,一个通信过程时延可分为处理时延,传播时延,排队时延,三者共同构筑了内存墙。其中处理时延最核心也最无解,先看它。 看个作为标准存储单元的双稳态触发器(至少教科书上就这…

2.1 Linux C 编程

一、Hello World 1、在用户根目录下创建一个C_Program,并在这里面创建3.1文件夹来保存Hellow World程序; 2、安装最新版nvim ①sudo apt-get install ninja-build gettext cmake unzip curl ②sudo apt install lua5.1 ③git clone https://github.…

深入理解Go语言GC机制

1、Go 1.3之前的标记-清除(mark and sweep)算法 Go 1.3之前的时候主要用的是普通的标记-清除算法,此算法主要由两个主要的步骤: 标记(Mark phase)清除(Sweep phase) 1&#xff09…

geemap学习笔记019:监督分类与精度验证(上)

前言 上一节中介绍了非监督分类,今天就详细介绍一下监督分类与精度验证。从这一节开始,我也是配置了本地的geemap,就可以不用colab了,配置也是花了挺长时间,但好在也是能够成功应用了,准备用两节的时间介绍…

【备忘干货】c/c++ (wasm)和js互相调用记录

c/c(wasm)和js互相调用记录 废话 :)准备工作:安装Emscripten初探:C(wasm)之hello world进一步探究:接口调用1.js调用c,一些基本类型的传递(char*,int,float)以…

博客访问量到达2万了!

博客访问量到达2万了!这也发生的太快了吧,前两天才1万7千访问量,用了平台送的1500的流量券,粉丝从1个(N年前的),蹭蹭的往上涨,这也太“假”了吧。关键我也是个菜鸟自学者&#xff0c…

前端组件库开发

通常我们会使用很多组件库,有时候我们会去看源码比如element,antd,然后发现多少是按需导出,和vue.use全局注册,依赖于框架的拓展。 组件库的开发依赖框架的版本和node的版本,这个是需要说明的,然…

Redis--15--缓存穿透 击穿 雪崩

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 缓存穿透 击穿 雪崩运行速度:1 缓存穿透问题描述:如何解决: 2 缓存击穿问题描述:如何解决: 3 缓存雪崩说明:解决方案: 缓存穿透 击穿 雪崩 问题描述: 由于海量的用…

BurpSuite 请求/响应解密插件开发

BurpSuite 请求/响应解密插件开发 本文主要记录如何利用burp官方的新版API即MontoyaApi 写一个请求/响应的解密插件。背景下面是主要的操作步骤:根据上述操作做完之后,生成,然后在burp中加载插件,然后通关抓包看效果,具…

VSCode 中将头文件和头文件函数分离,编译主函数跳出 undefined reference to 的问题解决

VSCode 编写 C (.h,.cpp 文件分离)代码,编写完成后,编译遇到了编译错误 undefined reference to xxx。 开始还以为使用了 -stdc20 而不能使用 #include “xxx.h" 方式头文件,但仔细一想虽然引入了 im…

Java 不要在父类的构造方法里面调用可以被子类重写的方法

不要在父类的构造方法(代码块)里面调用可以被子类重写的方法 我们从第一天学习Java开始,就对Java的类初始化顺序牢记于心。但是在实际开发过程中,似乎很难能接触这一部分的应用。在这之前,我也认为它只是面试中八股文而已,直到最…

版本依赖冲突问题排查过程记录

问题 开发平台在集成minio时&#xff0c;pom引入了sdk。 <dependency><groupId>io.minio</groupId><artifactId>minio</artifactId><version>8.5.7</version> </dependency>在调用上传文件API时&#xff0c;控制台报错&…

JDK1.8_X64在LINUX下安装

JDK1.8在LINUX下安装步骤&#xff1a; 在/usr/lib/目录下新建jvm文件夹&#xff0c;如果已有jvm文件夹&#xff0c;则将之前的JDK版本删除&#xff0c;即在jvm目录下执行命令&#xff1a;rm–rf *将JDK文件jdk-8u40-linux-x64.gz拷贝到/home/目录下&#xff1b;在/home/目录下…

探索JavaScript BOM:了解浏览器的内部机制和强大的API

​&#x1f308;个人主页&#xff1a;前端青山 &#x1f525;系列专栏&#xff1a;JavaScript篇 &#x1f516;人终将被年少不可得之物困其一生 依旧青山,本期给大家带来JavaScript篇专栏内容:JavaScript-BOM的概念和常用API BOM的概念和常用API BOM BOM&#xff08;Browser O…

免费网站快速收录工具,2023最新网站收录方法

在当今数字化时代&#xff0c;拥有一个被搜索引擎快速收录的网站对于个人、企业或机构而言至关重要。网站的快速收录意味着更广泛的曝光和更多的访问流量&#xff0c;这对于网络存在的任何实体都是非常有价值的。 网站快速收录的重要性 在庞大的互联网世界中&#xff0c;一切…

精神衰弱怎么办?如何改变精神衰弱?

精神衰弱也叫神经衰弱&#xff0c;跟个人体质有一定的关系&#xff0c;更多是因为环境因素的长期作用所致&#xff0c;比如长期处于紧张和压力之下&#xff0c;这容易产生精神衰弱。当下有个流行词叫“精神内耗”&#xff0c;这个严重的精神内耗其实同样会导致精神衰弱。 精神…

使用Prometheus监控Padavan路由器

Prometheus监控Padavan路由器 1、背景 近期在Synology&#xff08;群辉&#xff09;中安装一套Prometheus监控程序&#xff0c;目前已经监控Synology&#xff0c;然后家中有有路由器&#xff08;Padavan&#xff09;型号&#xff0c;也准备使用PrometheusGrafan进行监控。 ‍…

计算机网络扫盲(4)——时延

一、概述 在这里&#xff0c;我们考虑分组交换网的情况&#xff0c;因特网可以被看成是一种基础设施&#xff0c;该基础设施为运行在端系统上的分布式应用提供服务。在理想情况下&#xff0c;我们希望因特网服务能够在任意两个端系统之间随心所欲地移动数据而没有任何数据地丢失…

智能优化算法应用:基于海洋捕食者算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于海洋捕食者算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于海洋捕食者算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.海洋捕食者算法4.实验参数设定5.算法结果…