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

系列文章

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 值,而不会出现任何问题。对于其他一些转换,如将浮点类型转换为整型,则需要更加小心。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

实验案例二:多表查询

1、表联接类型。 表联接类型可以分为内联接.外联接和交叉联接等。 1.内联接。 内联接〈 inner join)是最常用的-一-种联接方式,只返回两个数据集合之间匹配关系的行,将位于两个互相交叉的数据集合中重叠部分以内的数…

Flink核心概念

并行度 当要处理的数据量非常大时,我们可以把一个算子操作,“复制”多份到多个节点,数据来了之后就可以到其中任意一个执行。这样一来,一个算子任务就被拆分成了多个并行的“子任务”(subtasks)&#xff0…

国标GB28181视频监控EasyCVR内网环境部署无法启动怎么办?

安防视频监控系统EasyCVR平台可拓展性强、视频能力灵活、部署轻快,可支持的主流标准协议有国标GB28181、RTSP/Onvif、RTMP等,以及支持厂家私有协议与SDK接入,包括海康Ehome、海大宇等设备的SDK等,能对外分发RTMP、RTSP、HTTP-FLV、…

U-Shape Transformer for Underwater Image Enhancement(用于水下图像增强的U型Transformer)总结

背景 现有的水下数据集或多或少存在图像数量少、水下场景少、甚至不是真实场景等缺点,限制了数据驱动的水下图像增强方法的性能。此外,水下图像在不同颜色通道和空间区域的衰减不一致也没有统一的框架。 贡献 1)提出了一种处理 UIE 任务的…

盘点2023年有哪些办公的效率工具

大家在使用Office时,会经常遇到一些比较繁杂的场景,比如设置段落格式,设置对齐方式,公式计算、文章排版等。使用工具能帮助我们轻松提高效率完成想要的效果,今天给大家介绍几款超实用的Office插件,不分分后…

uniapp 云打包 生成安卓证书文件

现在使用uniapp来开发小程序,H5,APP越来越多了,目前开发了一款APP,使用的也是uniapp。在此记录下用uniapp开发app云打包时约到的一些问题吧。 前因是我司安卓同学休产假,像云打包时需要的证书文件只能自己动手来搞。看…

国标GB28181安防监控平台EasyCVR录像时间轴优化步骤

视频云存储/安防监控EasyCVR视频汇聚平台基于云边端智能协同,支持海量视频的轻量化接入与汇聚、转码与处理、全网智能分发、视频集中存储等。音视频流媒体视频平台EasyCVR拓展性强,视频能力丰富,具体可实现视频监控直播、视频轮播、视频录像、…

SQL自学通之表达式条件语句与运算

目录 一、目标 二、表达式条件语句 1、表达式: 2、条件 2.1、WHERE 子句 三、运算 1、数值型运算: 1.1、加法() 1.2、减法 (-) 1.3、除法(/) 1.4、乘法 (*) 1.5、取模 (%) 优先级别…

第1章 理解知识图谱:知识图谱现状、知识图谱应用场景(二)

💗💗💗欢迎来到我的博客,你将找到有关如何使用技术解决问题的文章,也会找到某个技术的学习路线。无论你是何种职业,我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章,也欢…

GPT市场将取代插件商店 openAI已经关闭plugins申请,全部集成到GPTs(Actions)来连接现实世界,可以与物理世界互动了。

Actions使用了plugins的许多核心思想,也增加了新的特性。 ChatGPT的"Actions"与"Plugins"是OpenAI在GPT模型中引入的两种不同的功能扩展机制。这两种机制的目的是增强模型的功能,使其能够处理更多样化的任务和请求。下面是对两者的比…

在OSPF中使用基本ACL过滤路由信息示例

1、ACL的基本原理。 ACL由一系列规则组成,通过将报文与ACL规则进行匹配,设备可以过滤出特定的报文。设备支持软件ACL和硬件ACL两种实现方式。 2、ACL的组成。 ACL名称:通过名称来标识ACL,就像用域名代替IP地址一样,更…

LeetCode 每日一题 Day 3||深度优先搜索(DFS)

1038. 从二叉搜索树到更大和树 给定一个二叉搜索树 root (BST),请将它的每个节点的值替换成树中大于或者等于该节点值的所有节点值之和。 提醒一下, 二叉搜索树 满足下列约束条件: 节点的左子树仅包含键 小于 节点键的节点。节点的右子树仅…

ffmpeg编译支持AVS3编解码

libuavs3d ffmpeg的官方源码中已经支持了libuavs3d解码器的接口(libavcodec/libuavs3d.c中定义),因此如果需要编译ffmpeg支持libuavs3d解码器,只需要安装libuavs3d.so以及开启ffmpeg的编译选项即可。 安装libuavs3d解码器 #代码仓…

ssm党务政务服务热线平台源码和论文答辩PPT

摘要 首先,论文一开始便是清楚的论述了系统的研究内容。其次,剖析系统需求分析,弄明白“做什么”,分析包括业务分析和业务流程的分析以及用例分析,更进一步明确系统的需求。然后在明白了系统的需求基础上需要进一步地设计系统,主要包罗软件架构模式、整体功能模块、数据库设计…

生命周期模型构建方法与分析及实际案例应用

生命周期分析 (Life Cycle Analysis, LCA) 是评价一个产品系统生命周期整个阶段——从原材料的提取和加工,到产品生产、包装、市场营销、使用、再使用和产品维护,直至再循环和最终废物处置——的环境影响的工具。这种方法被认为是一种“从摇篮到坟墓”的…

MES管理系统与MOM系统分别有什么作用

随着科技的不断进步和全球化的竞争加剧,制造业正面临着前所未有的挑战和机遇。为了更好地适应市场需求,提高生产效率和质量,制造企业纷纷引进各种信息化管理系统。其中,MES管理系统和MOM系统是两种重要的解决方案。本文将详细探讨…

js中继承的方法

前言: 本人刚写了一篇原型链的封装继承多态,用家有儿女做的demo。其实我个人感觉封装和多态都容易去理解与实现。关键在于继承,js的才是比较难的,也容易让人混乱,至少我是因为继承头大过\(^o^)/~ js中有很多方法可以实现继承,这篇文章主要对继承的方法进行学习与测试。 这里…

c++ 三目运算符在类中的使用

简介 在类比较方面&#xff0c;三目运算符可以用于重载比较运算符。 代码示例1 #include <iostream> #include <cstring>class Person { public:Person(const char* name, int age) : m_age(age) {m_name new char[strlen(name) 1];strcpy(m_name, name);}~Pe…

LeetCode - 110. 平衡二叉树(C语言,二叉树,配图,简单)

根据题意&#xff0c;我们只需要比较当前节点的左右子树高度差是否小于1&#xff0c;利用分治法&#xff0c;只需要满足&#xff1a; 1. 根节点的左右子树的高度差小于1。 2. 根节点左右子树的满足高度差小于1&#xff0c;在往下走&#xff0c;判断左子树根节点的左右子树是否满…