百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

C|副作用,顺序点,及逻辑、逗号运算符中的顺序点

liebian365 2025-03-07 20:35 9 浏览 0 评论

对于以下语句:

y = (4 + x++) + (6 + x++);

编译器无法定义正确的行为。为什么?需了解C语言定义的副作用、顺序点。

1 副作用和顺序点

副作用(side effect)指的是在计算表达式时对某些东西(如存储在变量中的值)进行了修改。

顺序点(sequence point)是程序执行过程中的一个点,且要求在进入下一步之前将确保对所有的副作用都进行了评估。

在C中,语句中的分号就是一个顺序点,这意味着程序处理下一条语句之前,赋值运算符、递增运算符和递减运算符执行的所有修改都必须完成。任何完整的表达式末尾也都有一个顺序点,如用作while循环中检测条件的表达式:

while(guests++ <10)  //表达式guests++ <10就是一个完整表达式
    cout << guests <<endl;

回过头来再看下面的语句:

y = (4 + x++) + (6 + x++);

表达式4+x++不是一个完整表达式,因此,C不保证x的值在计算子表达式4+x++后立即增加1。在这个例子中,整条赋值语句是一个完整表达式,而分号标示了顺序点,因此C只保证程序执行到下一条语句之前,x的值将被递增两次。C没有规定是在计算每个子表达式之后将x的值递增,还是在整个表达式计算完毕后才将x的值递增,有鉴于此,您应避免使用这样的表达式。

2 逗号运算符是一个顺序点

语句块(两个大括号{}括住的语句)允许把两条或更多条语句放到按C句法只能放一条语句的地方。逗号运算符对表达式完成同样的任务,允许将两个表达式放到C句法只允许放在一个表达式的地方。

++j, --i // tow expressions count as one for syntax purposes

如下示例:

#include 
#include 
int main()
{
     using namespace std;
     cout << enter a word: string word cin>> word;
     char tmp;
     int i, j;
     for(j=0, i=word.size()-1; j<i; --i,++j)
     {
          tmp = word[i];
          word[i] = word[j];
          word[j] = tmp;
     }
     cout << word <<endl;
     return 0;
}

逗号并不总是逗号运算符。例如,下面这个声明中的逗号将变量列表中相邻的名称分开:

int i, j; // comma is a separator here, not an operator

逗号运算符最常见的用途是将两个或更多的表达式放到一个for循环表达式中。不过C还为这个运算符提供了另外两个特性。首先,它确保先计算第一个表达式,然后计算第二个表达式(换句话说,逗号运算符是一个顺序点),如下所示的表达式是最安全的:

i = 20, j = 2 * i; // i set to 20, then j set to 40

其次,C规定,逗号表达式的值是第二部分的值。例如,上述表达式的值为40,因为j = 2*i的值为40。

在所有运算符中,逗号运算符的优先级是最低的。例如,下面的语句:

cats = 17,24;

被解释为:

(cats = 17),240;

也就是说,将cats设置为17,240不起作用。然而,由于括号的优先级最高,下面的表达式将把cats设置为240---逗号右侧的表达式值:

cats = (17,240);

3 逻辑运算符中的顺序点

C规定,||运算符是个顺序点。也就是说,先修改左侧的值,再对右侧进行判定。

i++ < 6 || i ==j

假设i原来的值为10,则 在对i和j进行比较时,i的值将会为11。另外,如果左侧的表达式为true,则C将不会去判断右侧的表达式,因为只要一个表达式为true,则整个逻辑表达式为true。

和||运算符一样,&&运算符也是顺序点,因此将首先判断左侧,并且在右侧被判断之前产生所有的副作用。如果左侧为false,则整个逻辑表达式必定为false,在这种情况下,C++将不会再对右侧进行判定。

4 循环语句控制结构部分的顺序点

对于while或for循环,控制结构写在小括号内,并不需要以分号结尾。其后紧跟循环体,循环体可以是一条语句,或以两个大括号括住的语句块。如果在控制结构后以分号结束语句,则循环体只是一个空语句,其后的语句块可能并不是预料中的执行逻辑:

int i = 0;
while(name[i] != '\0'); // problem semicolon
{
     cout << name[i] << endl;
     i++;
}
cout << "Done!\n";

如以上的代码,就会陷入到一个死循环。

-End-

相关推荐

C++零基础入门学习指南(中篇)

目标:像拼装乐高一样理解程序模块,掌握内存管理核心技能...

“5 分钟 CMake 使用指南,解决我的 C++ 打包问题!”

...

Linux下跨语言调用C++实践

不同的开发语言适合不同的领域,例如Python适合做数据分析,C++适合做系统的底层开发,假如它们需要用到相同功能的基础组件,组件使用多种语言分别开发的话,不仅增加了开发和维护成本,而且不能确保多种语...

输入格式控制:C++程序中的数据接收与处理技巧

在C++编程中,输入输出是非常基本且重要的操作。尤其是输入部分,程序员通常需要从用户那里获取数据,并根据不同的输入格式进行处理。然而,用户的输入往往是多样化的,如何有效地控制输入格式,确保程序正确接收...

常见读写excel文件的库/类

在C++语言中读写EXCEL表格,有这几种方法:COM方式、ODBC方式、OLE方式、纯底层格式分析方式。Basicexcel使用方法:https://www.cnblogs.com/paullam/...

C++文档识别接口如何实现 高效办公

  数字化信息爆炸时代,办公效率的提升成为企业和个人的迫切需求。人工智能技术的飞速发展,为我们带来了前所未有的便利,翔云文档识别接口便是其中之一。  与传统的人工手动录入相比,文档识别接口优势显著。人...

C++如何生成Microsoft Word文档

...

超实用C++学习指南:语法要点、经典书籍、实战案例全汇总!

以下是为您整理的C++学习指南,综合了语法要点、资源推荐及实战方向,结合搜索结果和经典知识体系,帮助您系统学习:一、C++基础语法学习指南1.核心概念oC++是静态类型、编译式语言,支持面向对象和...

掌握C++文件读写,让代码更灵动!

文章改写指令通常涉及对原有文本进行调整、重组或重新表达,以保持或增强信息的准确性和可读性,同时可能改变风格、语气或目标受众。以下是一些具体的文章改写指令示例:·2.简化语言:→指令:将文章中的复杂词汇...

闲置宽带能换钱?P2P CDN、无线宝、赚钱宝到底靠不靠谱

无线宝类产品其实由来已久,无线宝类产品即与支付宝、余额宝、余利宝等货币基金毫无干系,与区块链“挖币”更存在本质的不同,而是一种利用家庭中的闲置宽带,通过流量来换取佣金的产品。无线宝类产品其实在过去几年...

攻略什么?闲置宽带还可以赚钱?

现在很多朋友在使用10Mbps、50Mbps甚至100Mbps的高速宽带,不过普通用户并不是长时间都需要这么高速的宽带。比如对于100Mbps的宽带用户,在日常浏览网页时,基本上2Mbps左右的带宽即...

明日学业水平考试开始报名 详细步骤都在这里

点击上面蓝字关注我们哦~日前,山东省教育考试院发布了《山东省2019年夏季学业水平考试报名考生操作说明》(点击文末阅读原文查看),明天就到了报名的时候了,详细的报名步骤、网上缴费流程、追加报考科目等...

瞄准用户上传带宽:HiWiFi 极路由 联合 迅雷 推出 “极赚钱”套餐

上次总理谈到宽带降价问题时,很多网友除了吐槽网速慢费用贵,还反映宽带网络的上下行速度不对等。比如说以前ADSL2M的宽带只有512Kbps的上行速度,现在升级到光纤网络之后,按理说技术上实现上下行...

揭秘P2P平台刷数据:交易额从100万到1200万

(作者:峰岭、刘珺、周娜)从默默无闻到万众瞩目,从“零数据”到“大数据”,从小众投资到大众理财,从个人借贷到企业借款,从个人信用到车、房、资产抵押……近两年来P2P行业以迅雷之速快速爆发,P2P平台也...

运营商让我签这个宽带违规使用告知函,我懵逼了

特么的是爱奇艺迅雷自己上传的p2p数据,btpt也会上传,直播也会上传,监控也会,传文件也会,到底他么的运营商你要干个啥啊,我不仅没捞着一分好处,夹在中间两头受气!真特么晦气这特么是谁弄的函?完全没搞...

取消回复欢迎 发表评论: